mand-mobile/site/theme/default/components/Doc.vue

655 lines
22 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<template>
<div class="mfe-blog-theme-default-doc doc-template">
<div class="default-doc-content">
<!-- 文档标题和描述front-matter -->
<div class="doc-content-top">
<p class="doc-content-title" v-if="info.title" v-html="info.title"></p>
<p class="doc-content-qrcode"
v-if="info.preview"
@mouseover="isQrcodeShow = true"
@mouseleave="isQrcodeShow = false">
<i class="icon-qr-code" :class="{active: isQrcodeShow}"></i>
<transition name="slide-fade">
<span class="qrcode-box" v-show="isQrcodeShow">
<i v-if="lang === 'en-US'">Scan QR code to preview</i>
<i v-else>扫码预览</i>
<qr-code :text="info.preview"></qr-code>
</span>
</transition>
</p>
</div>
<div class="doc-content-describe" v-if="info.describe" v-html="info.describe"></div>
<!-- 文档头部信息,介绍和引入 -->
<div class="doc-content-paragraph head" v-html="bodyHead"></div>
<!-- 文档内嵌Demo -->
<template v-if="demos && demos.length">
<div class="default-doc-demo-container">
<div class="default-doc-demo-list">
<div class="default-doc-demo"
v-for="(demo, index) in demos"
v-if="index % 2 === 0"
:key="index">
<div class="doc-demo-box"
:class="[
`doc-demo-box-${index}`,
demoBoxShowStat[index] ? 'active' : ''
]"
>
<div class="doc-demo-box-info">
<template v-if="lang === 'en-US'">
<h4 class="doc-demo-title"
v-html="demo.component.titleEnUS || demo.component.title || 'Basic'"
></h4>
<h5 class="doc-demo-describe"
v-if="demo.component.describeEnUS || demo.component.describe"
v-html="demo.component.describeEnUS || demo.component.describe"
></h5>
<h5 class="doc-demo-message"
v-if="demo.component.messageEnUS || demo.component.message"
v-html="demo.component.messageEnUS || demo.component.message"
></h5>
</template>
<template v-else>
<h4 class="doc-demo-title" v-html="demo.component.title || '基本'"></h4>
<h5 class="doc-demo-describe" v-if="demo.component.describe" v-html="demo.component.describe"></h5>
<h5 class="doc-demo-message" v-if="demo.component.message" v-html="demo.component.message"></h5>
</template>
</div>
<div class="doc-demo-box-preview">
<div class="doc-demo-box-preview-box" :style="{minHeight: `${demo.component.height}px`}">
<component :is="demo.component"></component>
</div>
</div>
<div class="doc-demo-box-code">
<div class="doc-demo-box-code-operate">
<i class="icon-hollowError"
@click="toggleDemoBox(index)"
></i>
<template v-if="lang === 'en-US'">
<i :class="isCopySuccess ? 'icon-question' : 'icon-paper'"
v-tooltip="{content: isCopySuccess ? 'Copied' : 'Copy Code', offset: 5}"
v-clipboard:copy="decodeURI(demo.raw)"
v-clipboard:success="onCopySuccess"></i>
<i class="demo-codesandbox"
v-if="demo.component.codeSandBox"
v-tooltip="{content: 'Open in CodeSandBox', offset: 5}">
<a :href="demo.component.codeSandBox" target="_blank"></a>
</i>
<i class="icon-edit"
v-tooltip="{content: 'Edit this page on Github', offset: 5}"
@click="goToDemo(index)"
></i>
</template>
<template v-else>
<i :class="isCopySuccess ? 'icon-question' : 'icon-paper'"
v-tooltip="{content: isCopySuccess ? '复制代码成功' : '复制代码', offset: 5}"
v-clipboard:copy="decodeURI(demo.raw)"
v-clipboard:success="onCopySuccess"></i>
<i class="demo-codesandbox"
v-if="demo.component.codeSandBox"
v-tooltip="{content: '在CodeSandBox打开', offset: 5}">
<a :href="demo.component.codeSandBox" target="_blank"></a>
</i>
<i class="icon-edit"
v-tooltip="{content: '在Github上编辑此页', offset: 5}"
@click="goToDemo(index)"
></i>
</template>
</div>
<pre>
<code class="lang-vue" v-html="demo.code"></code>
</pre>
</div>
<div class="doc-demo-box-toggle" @click="toggleDemoBox(index)">
<template v-if="demoBoxShowStat[index]">
<i class="icon-triangle-up"></i>
<span v-if="lang === 'en-US'">Hide Code</span>
<span v-else>代码收起</span>
</template>
<template v-else>
<i class="icon-triangle-down"></i>
<span v-if="lang === 'en-US'">Show Code</span>
<span v-else>代码展示</span>
</template>
</div>
</div>
</div>
</div>
<div class="default-doc-demo-list">
<div class="default-doc-demo"
v-for="(demo, index) in demos"
v-if="index % 2 !== 0"
:key="index">
<div class="doc-demo-box"
:class="[
`doc-demo-box-${index}`,
demoBoxShowStat[index] ? 'active' : ''
]"
>
<div class="doc-demo-box-info">
<template v-if="lang === 'en-US'">
<h4 class="doc-demo-title"
v-html="demo.component.titleEnUS || demo.component.title || 'Basic'"
></h4>
<h5 class="doc-demo-describe"
v-if="demo.component.describeEnUS || demo.component.describe"
v-html="demo.component.describeEnUS || demo.component.describe"
></h5>
<h5 class="doc-demo-message"
v-if="demo.component.messageEnUS || demo.component.message"
v-html="demo.component.messageEnUS || demo.component.message"
></h5>
</template>
<template v-else>
<h4 class="doc-demo-title" v-html="demo.component.title || '基本'"></h4>
<h5 class="doc-demo-describe" v-if="demo.component.describe" v-html="demo.component.describe"></h5>
<h5 class="doc-demo-message" v-if="demo.component.message" v-html="demo.component.message"></h5>
</template>
</div>
<div class="doc-demo-box-preview">
<div class="doc-demo-box-preview-box" :style="{minHeight: `${demo.component.height}px`}">
<component :is="demo.component"></component>
</div>
</div>
<div class="doc-demo-box-code">
<div class="doc-demo-box-code-operate">
<i class="icon-hollowError"
@click="toggleDemoBox(index)"
></i>
<template v-if="lang === 'en-US'">
<i :class="isCopySuccess ? 'icon-question' : 'icon-paper'"
v-tooltip="{content: isCopySuccess ? 'Copied' : 'Copy Code', offset: 5}"
v-clipboard:copy="decodeURI(demo.raw)"
v-clipboard:success="onCopySuccess"></i>
<i class="demo-codesandbox"
v-if="demo.component.codeSandBox"
v-tooltip="{content: 'Open in CodeSandBox', offset: 5}">
<a :href="demo.component.codeSandBox" target="_blank"></a>
</i>
<i class="icon-edit"
v-tooltip="{content: 'Edit this page on Github', offset: 5}"
@click="goToDemo(index)"
></i>
</template>
<template v-else>
<i :class="isCopySuccess ? 'icon-question' : 'icon-paper'"
v-tooltip="{content: isCopySuccess ? '复制代码成功' : '复制代码', offset: 5}"
v-clipboard:copy="decodeURI(demo.raw)"
v-clipboard:success="onCopySuccess"></i>
<i class="demo-codesandbox"
v-if="demo.component.codeSandBox"
v-tooltip="{content: '在CodeSandBox打开', offset: 5}">
<a :href="demo.component.codeSandBox" target="_blank"></a>
</i>
<i class="icon-edit"
v-tooltip="{content: '在Github上编辑此页', offset: 5}"
@click="goToDemo(index)"
></i>
</template>
</div>
<pre>
<code class="lang-vue" v-html="demo.code"></code>
</pre>
</div>
<div class="doc-demo-box-toggle" @click="toggleDemoBox(index)">
<template v-if="demoBoxShowStat[index]">
<i class="icon-triangle-up"></i>
<span v-if="lang === 'en-US'">Hide Code</span>
<span v-else>代码收起</span>
</template>
<template v-else>
<i class="icon-triangle-down"></i>
<span v-if="lang === 'en-US'">Show Code</span>
<span v-else>代码展示</span>
</template>
</div>
</div>
</div>
</div>
</div>
</template>
<!-- 文档底部信息API -->
<div class="doc-content-paragraph tail"
v-if="bodyTail"
v-html="bodyTail">
</div>
<div class="doc-content-bottom">
<router-link class="prev" :to="prevRoute.path" v-if="prevRoute">
<i>Prev</i>
<p v-html="prevRoute.meta.text"></p>
</router-link>
<router-link class="next" :to="nextRoute.path" v-if="nextRoute">
<i>Next</i>
<p v-html="nextRoute.meta.text"></p>
</router-link>
</div>
</div>
<!-- 文档目录索引 -->
<div
v-if="!hiddenToc"
class="default-doc-toc"
:class="{'is-stricky': isTocStricky}"
v-html="toc">
</div>
</div>
</template>
<script>
import VueQRCodeComponent from 'vue-qrcode-component'
export default {
components: {
'qr-code': VueQRCodeComponent,
},
props: ['info', 'body', 'toc', 'demos'],
data() {
return {
demoBoxShowStat: [],
activeDemoBoxZoonPos: {},
isTocStricky: false,
isQrcodeShow: false,
isCopySuccess: false,
}
},
computed: {
bodyHead() {
return this.body.split('<!-- DEMO -->')[0]
},
bodyTail() {
return this.body.split('<!-- DEMO -->')[1]
},
demoBox() {
return $('.doc-demo-box')
},
previewBox() {
return $('.doc-demo-box-preview')
},
codeBox() {
return $('.doc-demo-box-code')
},
curRouteIndex() {
return this.$route.meta.index
},
prevRoute() {
return this.findRoute(this.curRouteIndex - 1, -1)
},
nextRoute() {
return this.findRoute(this.curRouteIndex + 1, 1)
},
lang() {
return ~this.$route.path.indexOf('zh-CN') ? 'zh-CN' : 'en-US'
},
hiddenToc() {
return this.info.toc === 'hidden'
}
},
mounted() {
if (!this.hiddenToc) {
const scrollTop = document.body.scrollTop || document.documentElement.scrollTop
$(window).bind('scroll', () => {
const scrollTop = document.body.scrollTop || document.documentElement.scrollTop
this.strickyTocBar(scrollTop)
// this.strickyToggleBar(scrollTop)
})
this.strickyTocBar(scrollTop)
}
if (location.hash) {
const tmpHash = location.hash.substr(1)
location.hash = ''
location.hash = tmpHash
}
},
methods: {
findRoute(start, direction = 1) {
const routes = window.$routes[this.lang]
while (start >= 0 && start <= routes.length - 1 &&
((!routes[start].meta.src && !routes[start].meta.markdown) || routes[start].redirect)) {
start += direction
}
return routes[start]
},
toggleDemoBox(index, event) {
const demoBox = $(`.doc-demo-box-${index}`)
const toggleBar = demoBox.find('.doc-demo-box-toggle')
const stat = !this.demoBoxShowStat[index]
if (stat) {
toggleBar.addClass('active')
// this.toggleStrickyToggleBar()
} else {
toggleBar.removeClass('active')
// this.toggleStrickyToggleBar()
}
this.$set(this.demoBoxShowStat, index, stat)
},
strickyTocBar(scrollTop) {
window.requestAnimationFrame(() => {
if (scrollTop > 96) {
this.isTocStricky = true
} else {
this.isTocStricky = false
}
})
},
strickyToggleBar (scrollTop) {
window.requestAnimationFrame(() => {
this.toggleStrickyToggleBar(scrollTop)
})
},
toggleStrickyToggleBar (scrollTop) {
scrollTop = scrollTop || document.body.scrollTop || document.documentElement.scrollTop
$.each($('.doc-demo-box-toggle'), (index, item) => {
const width = $(item).width()
const height = $(item).height()
const demo = $(item).siblings('.doc-demo-box-code')
const offset = demo.offset()
const codeHeight = demo.height()
const top = $(window).height() - (offset.top - scrollTop)
const bottom = $(window).height() - (offset.top + codeHeight - scrollTop)
if ($(item).hasClass('active') && top >= 0 && bottom <= 0) {
if (!$(item).hasClass('is-stricky')) {
$(item).css({maxWidth: `${width}px`, left: `${$(item).offset().left}px`})
$(item).addClass('is-stricky')
}
} else {
$(item).css({maxWidth: `${width}px`, left: '0px'})
$(item).removeClass('is-stricky')
}
})
},
goToDemo (index) {
const component = this.info.preview.split('#')[1]
if (component) {
window.open(`https://github.com/didi/mand-mobile/edit/master/components/${component}/demo/cases/demo${index}.vue`)
}
console.log(this.info.preview, index)
},
onCopySuccess (e) {
this.isCopySuccess = true
setTimeout(() => {
this.isCopySuccess = false
}, 1000)
}
},
}
</script>
<style lang="stylus">
.mfe-blog-theme-default-doc
position relative
block()
padding-right 12%
border-left solid 1px #e8e8e8
box-sizing border-box
.doc-content-top, .doc-content-describe, .default-doc-content, .doc-content-paragraph
block()
box-sizing border-box
.doc-content-top
margin-bottom 20px
.doc-content-title
font-size 28px
font-weight 500
color #1f2f3d
.doc-content-title, .doc-content-qrcode
float left
line-height 1
.doc-content-qrcode
position relative
margin-top 4px
margin-left 10px
i.icon-qr-code
font-size 22px
color #999
&.active
color #3ca0e6
span
position absolute
left -61px
top 30px
z-index 2
width 150px
// height 150px
padding 10px 15px
box-sizing border-box
background #fff
box-shadow 0 4px 8px rgba(0, 0, 0, .08)
border-radius 4px
border solid 1px #f0f0f0
i
display inline-block
width 100%
text-align center
font-size 12px
color #999
font-style normal
.doc-content-describe
font-size 16px
font-weight 400
color #666
margin-top 20px
.doc-content-bottom
block()
position absolute
left 0
bottom 0
padding 20px 64px
box-sizing border-box
a
text-decoration none
i
color #999
font-size 12px
font-style normal
p
margin-top 5px
color #048EFA
font-size 14px
a.prev
float left
text-align left
a.next
float right
text-align right
.default-doc-content
position relative
min-height 800px
padding 0 64px 87px
.default-doc-demo-container
block()
.default-doc-demo-list
float left
width 49%
&:first-of-type
margin-right 2%
.default-doc-demo, .doc-demo-box-info, .doc-demo-title, .doc-demo-describe, .doc-demo-box
block()
box-sizing border-box
.default-doc-demo
margin-bottom 20px
.doc-demo-box-info
padding 15px
.doc-demo-title
display flex
align-items center
font-size 14px
font-weight 500
.doc-demo-describe
margin-top 10px
color #999
font-size 14px
font-weight 400
.doc-demo-message
display inline-block
width 100%
box-sizing border-box
border-left .3em solid #048EFA
padding 1em
margin-left 0
margin-top 10px
background rgba(252, 145, 83, 0.05)
border-radius 4px
font-weight 400
.doc-demo-box
position relative
padding-bottom 24px
border solid 1px #ebebeb
border-radius 2px
transition all .3s
overflow hidden
&:hover
// box-shadow 0 0 8px rgba(0, 0, 0, .08)
.doc-demo-box-toggle span
transform translateX(0)
&.active
.doc-demo-box-code, .doc-demo-box-toggle
display block
.demo-codesandbox
width 16px
height 16px
background url('') no-repeat
background-size contain
a
float left
width 100%
height 100%
margin 0
.doc-demo-box-preview
position relative
float left
width 100%
padding 10px 0
box-sizing border-box
border-top solid 1px #ebebeb
background #F9FAFB
.doc-demo-box-preview-box
position relative
width 100%
max-width 450px
margin 0 auto
overflow hidden
.md-example-child
zoom .6
ul>li
list-style none !important
.doc-demo-box-code
position relative
display none
width 100%
overflow hidden
box-sizing border-box
border-top dashed 1px #ebebeb
pre
margin-bottom 0
background #fff
transition all .3s
.doc-demo-box-toggle
// display none
position absolute
bottom 0
left 0
right 0
z-index 1102
width 100%
height 24px
border-top solid 1px #ebebeb
cursor pointer
text-align center
line-height 24px
font-size 12px
color #ccc
transition background .3s
background #fff
overflow hidden
i
font-size 24px
line-height 24px
span
position absolute
top 0
right 20px
transform translateX(200%)
transition transform .3s ease-in-out
font-weight 500
&:hover
background #fafafa
i, span
color #256fa3
&.is-stricky
position fixed
bottom 0
.doc-demo-box-code-operate
position absolute
top 0
right 0
z-index 100
padding 10px 0
cursor pointer
i
float right
margin-right 10px
font-size 16px
color #ccc
transition all .3s
&:hover
transform scale(1.2)
&:active, &:visited, &:focus
box-shadow none
outline none
@media (max-width: 1500px)
.doc-demo-box-preview-box
max-width 400px !important
.md-example-child
zoom .533 !important
@media (max-width: 1200px)
.doc-demo-box-preview-box
max-width 350px !important
.md-example-child
zoom .467 !important
@media (max-width: 1000px)
.default-doc-demo-list
width 100% !important
margin-right 0 !important
.doc-demo-box-preview-box
width 100% !important
.doc-demo-box-code
position static
float left
width 100%
border-left none !important
.mfe-blog-theme-default-doc
padding-right 0 !important
.default-doc-toc
display none
@media (max-width: 750px)
.mfe-blog-theme-default-doc
padding 0
.default-doc-content
padding 15px 15px 100px 15px !important
.doc-content-title
font-size 22px !important
.doc-content-qrcode i.icon-qr-code
font-size 16px !important
.doc-content-bottom
padding 20px 15px !important
</style>