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

655 lines
22 KiB
Vue
Raw Normal View History

2018-03-26 16:04:04 +08:00
<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">
2018-05-21 19:37:56 +08:00
<i v-if="lang === 'en-US'">Scan QR code to preview</i>
<i v-else>扫码预览</i>
2018-03-26 16:04:04 +08:00
<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">
2018-05-21 19:37:56 +08:00
<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>
2018-03-26 16:04:04 +08:00
</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>
2018-05-21 19:37:56 +08:00
<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>
2018-03-26 16:04:04 +08:00
<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]">
2018-09-12 20:48:33 +08:00
<i class="icon-triangle-up"></i>
2018-05-21 19:37:56 +08:00
<span v-if="lang === 'en-US'">Hide Code</span>
<span v-else>代码收起</span>
2018-03-26 16:04:04 +08:00
</template>
<template v-else>
2018-09-12 20:48:33 +08:00
<i class="icon-triangle-down"></i>
2018-05-21 19:37:56 +08:00
<span v-if="lang === 'en-US'">Show Code</span>
<span v-else>代码展示</span>
2018-03-26 16:04:04 +08:00
</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">
2018-05-21 19:37:56 +08:00
<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>
2018-03-26 16:04:04 +08:00
</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>
2018-05-21 19:37:56 +08:00
<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>
2018-03-26 16:04:04 +08:00
<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]">
2018-09-12 20:48:33 +08:00
<i class="icon-triangle-up"></i>
2018-05-21 19:37:56 +08:00
<span v-if="lang === 'en-US'">Hide Code</span>
<span v-else>代码收起</span>
2018-03-26 16:04:04 +08:00
</template>
<template v-else>
2018-09-12 20:48:33 +08:00
<i class="icon-triangle-down"></i>
2018-05-21 19:37:56 +08:00
<span v-if="lang === 'en-US'">Show Code</span>
<span v-else>代码展示</span>
2018-03-26 16:04:04 +08:00
</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
2018-06-21 16:37:22 +08:00
v-if="!hiddenToc"
2018-03-26 16:04:04 +08:00
class="default-doc-toc"
:class="{'is-stricky': isTocStricky}"
v-html="toc">
</div>
</div>
</template>
<script>
import VueQRCodeComponent from 'vue-qrcode-component'
import { setTimeout } from 'timers';
2018-03-26 16:04:04 +08:00
export default {
components: {
'qr-code': VueQRCodeComponent,
},
2018-05-21 19:37:56 +08:00
props: ['info', 'body', 'toc', 'demos'],
2018-03-26 16:04:04 +08:00
data() {
return {
demoBoxShowStat: [],
activeDemoBoxZoonPos: {},
isTocStricky: false,
isQrcodeShow: false,
isCopySuccess: false,
2018-03-26 16:04:04 +08:00
}
},
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)
2018-05-21 19:37:56 +08:00
},
lang() {
return ~this.$route.path.indexOf('zh-CN') ? 'zh-CN' : 'en-US'
},
2018-06-21 16:37:22 +08:00
hiddenToc() {
return this.info.toc === 'hidden'
}
2018-03-26 16:04:04 +08:00
},
mounted() {
2018-06-21 16:37:22 +08:00
if (!this.hiddenToc) {
2018-03-26 16:04:04 +08:00
const scrollTop = document.body.scrollTop || document.documentElement.scrollTop
2018-06-21 16:37:22 +08:00
$(window).bind('scroll', () => {
const scrollTop = document.body.scrollTop || document.documentElement.scrollTop
this.strickyTocBar(scrollTop)
// this.strickyToggleBar(scrollTop)
})
2018-03-26 16:04:04 +08:00
this.strickyTocBar(scrollTop)
2018-06-21 16:37:22 +08:00
}
2018-03-26 16:04:04 +08:00
if (location.hash) {
const tmpHash = location.hash.substr(1)
location.hash = ''
location.hash = tmpHash
}
},
methods: {
findRoute(start, direction = 1) {
2018-05-24 20:18:42 +08:00
const routes = window.$routes[this.lang]
2018-03-26 16:04:04 +08:00
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)
2018-03-26 16:04:04 +08:00
}
},
}
</script>
<style lang="stylus" scoped>
.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
2018-05-04 20:07:49 +08:00
font-weight 500
2018-03-26 16:04:04 +08:00
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
2018-04-26 14:22:52 +08:00
2018-03-26 16:04:04 +08:00
.doc-content-describe
font-size 16px
font-weight 400
color #666
margin-top 20px
2018-04-26 14:22:52 +08:00
2018-03-26 16:04:04 +08:00
.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
2018-03-26 16:04:04 +08:00
padding 0 64px 87px
2018-04-26 14:22:52 +08:00
2018-03-26 16:04:04 +08:00
.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
2018-09-12 20:48:33 +08:00
padding 15px
2018-03-26 16:04:04 +08:00
.doc-demo-title
2018-09-12 20:48:33 +08:00
display flex
align-items center
font-size 14px
2018-03-26 16:04:04 +08:00
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 44px
border solid 1px #ebebeb
border-radius 2px
transition all .3s
overflow hidden
&:hover
// box-shadow 0 0 8px rgba(0, 0, 0, .08)
2018-03-26 16:04:04 +08:00
.doc-demo-box-toggle span
transform translateX(0)
&.active
.doc-demo-box-code, .doc-demo-box-toggle
display block
2018-05-11 11:52:31 +08:00
.demo-codesandbox
width 16px
height 16px
background url('') no-repeat
2018-05-11 11:52:31 +08:00
background-size contain
a
float left
width 100%
height 100%
margin 0
2018-03-26 16:04:04 +08:00
.doc-demo-box-preview
position relative
float left
width 100%
padding 10px 0
box-sizing border-box
border-top solid 1px #ebebeb
2018-08-29 17:37:25 +08:00
background #f3f4f5
2018-03-26 16:04:04 +08:00
.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
2018-04-26 14:22:52 +08:00
position relative
2018-03-26 16:04:04 +08:00
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 44px
border-top solid 1px #ebebeb
cursor pointer
text-align center
line-height 44px
font-size 12px
color #ccc
transition background .3s
background #fff
overflow hidden
i
2018-09-12 20:48:33 +08:00
font-size 24px
2018-03-26 16:04:04 +08:00
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
2018-04-26 14:22:52 +08:00
2018-03-26 16:04:04 +08:00
@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
2018-04-26 14:22:52 +08:00
display none
2018-03-26 16:04:04 +08:00
@media (max-width: 750px)
.mfe-blog-theme-default-doc
padding 0
.default-doc-content
2018-05-04 20:07:49 +08:00
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
2018-03-26 16:04:04 +08:00
</style>