forked from opentiny/tiny-vue
190 lines
6.5 KiB
Vue
190 lines
6.5 KiB
Vue
<template>
|
||
<div class="wp100 hp100 f-r of-hidden">
|
||
<div class="w230 pt20 of-auto sm-hidden b-r bg-white" :class="{ 'fixed-menu': showFixedMenu }">
|
||
<tiny-tree-menu
|
||
class="!w213"
|
||
:data="menuData"
|
||
:filter-node-method="fn.searchMenu"
|
||
@current-change="fn.clickMenu"
|
||
></tiny-tree-menu>
|
||
</div>
|
||
<div class="fi-1 f-c px20 pb30 f-c pr200 of-auto">
|
||
<!-- 标题 -->
|
||
<div class="py20">
|
||
<component :is="state.currMd"></component>
|
||
</div>
|
||
<div id="preview" class="bg-white">
|
||
<div class="mb20 py10 pl16 child<code>p4 child<code>bg-lightless">
|
||
<div class="mr20 fw-bold">
|
||
{{ state.currDemo?.name['zh-CN'] }}( <span class="allselect">{{ state.currDemo?.codeFiles[0] }}</span
|
||
>):
|
||
</div>
|
||
<div v-html="state.currDemo?.desc['zh-CN']"></div>
|
||
</div>
|
||
<!-- 预览 -->
|
||
<div class="rel px20" :id="state.currDemo?.demoId">
|
||
<div class="phone-container" @dblclick="fn.openInVscode(state.currDemo)">
|
||
<div class="mobile-view-container">
|
||
<component :is="state.comp"></component>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<!-- API表格 -->
|
||
<div v-if="state.currApi?.length" class="mt20 f24 fw-bold">组件API</div>
|
||
<div v-for="(oneGroup, idx) in state.currApi" :key="idx">
|
||
<div class="mt20 f-r f-pos-start fw-bold">
|
||
<div :id="oneGroup.name" class="f18">
|
||
{{ oneGroup.name }}
|
||
</div>
|
||
<div class="ml12 b-a-primary c-primary px8 py4">
|
||
{{ oneGroup.type }}
|
||
</div>
|
||
</div>
|
||
<div v-for="(oneApiArr, key) in oneGroup" :key="key">
|
||
<div v-if="key !== 'name' && key !== 'type' && oneApiArr.length > 0">
|
||
<div class="f18 py28">
|
||
{{ key }}
|
||
</div>
|
||
<table class="api-table">
|
||
<thead>
|
||
<tr>
|
||
<th width="20%">名称</th>
|
||
<th width="15%">类型</th>
|
||
<th width="20%">默认值</th>
|
||
<th width="55%">说明</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
<tr v-for="row in oneApiArr" :key="row.name">
|
||
<td>
|
||
<a v-if="row.demoId" class="c-primary h:c-error cur-hand" @click="fn.selectDemo(row.demoId)">{{
|
||
row.name
|
||
}}</a>
|
||
<span v-else>{{ row.name }}</span>
|
||
</td>
|
||
<td>{{ row.type }}</td>
|
||
<td v-html="typeof row.defaultValue === 'string' ? row.defaultValue || '--' : row.defaultValue"></td>
|
||
<td v-html="row.desc['zh-CN']"></td>
|
||
</tr>
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<!-- 右边浮动所有的demos -->
|
||
<tiny-floatbar v-if="state.demos?.length > 0" class="!top120 !z1 !right25">
|
||
<div class="f12 ofy-auto">
|
||
<div
|
||
v-for="demo in state.demos"
|
||
:key="demo.demoId"
|
||
@click="fn.selectDemo(demo.demoId)"
|
||
class="w130 px10 py4 bg-light f-r f-pos-between"
|
||
:class="{ 'c-error': state.currDemo === demo }"
|
||
>
|
||
<div class="link-primary h:c-error h:td-under ellipsis">
|
||
{{ demo.name['zh-CN'] }}
|
||
<Icon-star-icon v-if="state.currDemo === demo" style="fill: #ee343f" />
|
||
</div>
|
||
<IconOpeninVscode @click.stop="fn.openInVscode(demo)" class="f18 cur-hand" />
|
||
</div>
|
||
</div>
|
||
</tiny-floatbar>
|
||
</div>
|
||
</template>
|
||
|
||
<script>
|
||
import { hooks } from '@opentiny/vue-common'
|
||
import { Floatbar, TreeMenu, Button, Tooltip } from '@opentiny/vue'
|
||
import { iconStarActive, iconSelect } from '@opentiny/vue-icon'
|
||
import { menuData, demos, demoStr, demoVue, mds } from './resourceMobile.js'
|
||
import { useModeCtx } from './uses'
|
||
import { getDemosConfig, getApisConfig } from './utils/componentsDoc'
|
||
|
||
export default {
|
||
props: {
|
||
showFixedMenu: Boolean
|
||
},
|
||
components: {
|
||
TinyFloatbar: Floatbar,
|
||
TinyTreeMenu: TreeMenu,
|
||
TinyButton: Button,
|
||
TinyTooltip: Tooltip,
|
||
IconStarIcon: iconStarActive(),
|
||
IconOpeninVscode: iconSelect()
|
||
},
|
||
setup() {
|
||
const { state: modeState, fn: modeFn } = useModeCtx()
|
||
const state = hooks.reactive({
|
||
demos: [], // 组件的所有示例
|
||
currDemo: null, // 选中的demo
|
||
currApi: [], // 当前path下的api
|
||
comp: null, // 当前示例的组件实例
|
||
currDemoSrc: '',
|
||
currMd: hooks.computed(() => mds[`${modeState.pathName}.cn.md`])
|
||
})
|
||
const fn = {
|
||
// 菜单搜索:忽略大小写
|
||
searchMenu: (value, data) => {
|
||
if (!value) return true
|
||
return data.label.toLowerCase().includes(value.toLowerCase())
|
||
},
|
||
// 点击菜单:如果是二级菜单,根据path 刷新整个页面内容
|
||
clickMenu: async (menu) => {
|
||
if (menu.nameCn && menu.key !== state.key) {
|
||
modeState.pathName = menu.key
|
||
await _switchPath()
|
||
}
|
||
},
|
||
// 点击示例
|
||
selectDemo: async (demoId) => {
|
||
const demo = state.demos.find((d) => d.demoId === demoId)
|
||
if (state.currDemo !== demo) {
|
||
state.currDemo = demo
|
||
await _switchDemo()
|
||
}
|
||
},
|
||
openInVscode: (demo) => {
|
||
fetch(`/__open-in-editor?file=../sites/demos/mobile/app/${modeState.pathName}/${demo.codeFiles[0]}`)
|
||
}
|
||
}
|
||
|
||
hooks.onMounted(() => {
|
||
_switchPath()
|
||
})
|
||
|
||
// 以下私有方法,无须传递给vue模板的。
|
||
async function _switchPath() {
|
||
const demosModule = demos[`../../sites/demos/mobile/app/${modeState.pathName}/webdoc/${modeState.pathName}.js`]
|
||
const demosConfig = await getDemosConfig(demosModule)
|
||
state.demos = demosConfig.demos
|
||
state.currDemo = state.demos.find((d) => d.demoId === modeState.demoId) || state.demos?.[0]
|
||
state.currApi = (await getApisConfig(modeState.pathName, 'mobile')).apis
|
||
await _switchDemo()
|
||
}
|
||
async function _switchDemo() {
|
||
modeState.demoId = state.currDemo.demoId
|
||
const path = `../../sites/demos/mobile/app/${modeState.pathName}/${state.currDemo?.codeFiles[0]}`
|
||
|
||
// 查找源码 查找组件
|
||
state.currDemoSrc = await demoStr[path]()
|
||
const comp = await demoVue[path]()
|
||
|
||
state.comp = hooks.markRaw(comp.default)
|
||
|
||
modeFn.cacheCtx()
|
||
modeFn.pushToUrl()
|
||
}
|
||
|
||
return {
|
||
menuData,
|
||
state,
|
||
fn,
|
||
modeState,
|
||
modeFn
|
||
}
|
||
}
|
||
}
|
||
</script>
|