mirror of
				https://gitee.com/vant-contrib/vant.git
				synced 2025-11-04 21:02:09 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			234 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			234 lines
		
	
	
		
			8.8 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="toogleSource"
 | 
						|
          />
 | 
						|
        </div>
 | 
						|
        <div
 | 
						|
          v-show="showSource"
 | 
						|
          v-html="unescape(codeSnippet)"
 | 
						|
          class="demo-playground--code--content"
 | 
						|
        ></div>
 | 
						|
      </div>
 | 
						|
    </template>
 | 
						|
  </div>
 | 
						|
</template>
 | 
						|
 | 
						|
<script>
 | 
						|
// from https://30secondsofcode.org
 | 
						|
function copyToClipboard(str) {
 | 
						|
  const el = document.createElement('textarea');
 | 
						|
  el.value = str;
 | 
						|
  el.setAttribute('readonly', '');
 | 
						|
  el.style.position = 'absolute';
 | 
						|
  el.style.left = '-9999px';
 | 
						|
  document.body.appendChild(el);
 | 
						|
 | 
						|
  const selection = document.getSelection();
 | 
						|
 | 
						|
  if (!selection) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  const selected = selection.rangeCount > 0 ? selection.getRangeAt(0) : false;
 | 
						|
 | 
						|
  el.select();
 | 
						|
  document.execCommand('copy');
 | 
						|
  document.body.removeChild(el);
 | 
						|
 | 
						|
  if (selected) {
 | 
						|
    selection.removeAllRanges();
 | 
						|
    selection.addRange(selected);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
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,
 | 
						|
    toogleSource() {
 | 
						|
      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>
 |