neverland 3dcc92a5c0
chore: bump prettier v3 and format all code (#12111)
* chore: bump prettier v3 and format all code

* chore: mjs config

* chore: revert

* chore: revert

* chore: update lock
2023-07-22 14:14:14 +08:00

211 lines
8.3 KiB
Vue

<template>
<div :class="{ 'demo-playground': !inline, transform }">
<slot v-if="inline" />
<template v-else>
<div class="demo-playground--previewer" :class="{ compact }">
<slot />
</div>
<div class="demo-playground--code">
<div class="demo-playground--code--actions">
<span></span>
<button
title="Copy source code"
class="action-icon"
role="copy"
:data-status="copyStatus"
@click="copySourceCode"
/>
<button
title="Toggle source code panel"
class="action-icon"
role="source"
@click="toggleSource"
/>
</div>
<div
v-show="showSource"
v-html="unescape(codeSnippet)"
class="demo-playground--code--content"
></div>
</div>
</template>
</div>
</template>
<script>
import { copyToClipboard } from '../../common';
export default {
name: 'DemoPlayground',
props: {
originCode: String, // 源文件内容
codeSnippet: String, // 源文件转 html 后的内容
transform: Boolean, // 防止 position fixed 内容飞出预览区域
compact: Boolean, // 紧凑模式
inline: Boolean, // 不需要容器
},
data() {
return {
showSource: false,
copyStatus: 'ready',
};
},
methods: {
unescape,
toggleSource() {
this.showSource = !this.showSource;
},
copySourceCode() {
copyToClipboard(unescape(this.originCode));
this.copyStatus = 'copied';
setTimeout(() => {
this.copyStatus = 'ready';
}, 2000);
},
},
};
</script>
<style lang="less" scoped>
.demo-playground {
background-color: #fff;
border: 1px solid #ebedf1;
border-radius: 1px;
margin: 24px 0;
&.transform {
transform: translate(0, 0);
}
&--previewer {
padding: 40px 24px;
border-bottom: 1px solid #ebedf1;
&.compact {
padding: 0;
}
}
&--code {
&--actions {
display: flex;
height: 40px;
padding: 0 1em;
align-items: center;
> a:not(:last-child),
> button:not(:last-child) {
margin-right: 8px;
}
> a {
display: flex;
}
button {
position: relative;
display: inline-block;
width: 16px;
height: 16px;
padding: 0;
border: 0;
box-sizing: border-box;
cursor: pointer;
opacity: 0.6;
outline: none;
transition:
opacity 0.2s,
background 0.2s;
// expand click area
&::after {
content: '';
position: absolute;
top: -8px;
left: -8px;
right: -8px;
bottom: -8px;
}
&:hover {
opacity: 0.8;
}
&:active {
opacity: 0.9;
}
&:disabled {
opacity: 0.2;
cursor: not-allowed;
}
&[role='codesandbox'] {
background-position: -18px 0;
}
&[role='codepen'] {
background-position: -36px 0;
}
&[role='source'] {
background-position: -72px 0;
}
&[role='change-jsx'] {
background-position: -90px 0;
}
&[role='change-tsx'] {
background-position: -108px 0;
}
&[role='open-demo'] {
background-position: -126px 0;
}
&[role='motions'] {
background-position: -162px 0;
}
&[role='sketch-component'] {
background-position: -182px 0;
}
&[role='sketch-group'] {
background-position: -200px 0;
}
&[role='copy'][data-status='ready'] {
background-position: -54px 0;
}
&[role='copy'][data-status='copied'] {
pointer-events: none;
background-position: -54px -16px;
}
&[role='refresh'] {
background-position-x: -144px;
}
}
// split action buttons by a blank node
> span {
flex: 1;
display: inline-block;
}
}
&--content {
border-top: 1px dashed #ebedf1;
:deep(pre) {
margin: 0;
}
:deep(.language-html) {
border-radius: 0;
}
}
}
}
.action-icon {
background: url('')
no-repeat 0 0/230px auto;
}
</style>