version: 4.6.2

This commit is contained in:
XiaoDaiGua-Ray 2024-01-31 17:21:41 +08:00
parent a00e3ca557
commit 25599cea24
27 changed files with 573 additions and 209 deletions

View File

@ -1,9 +1,6 @@
node_modules/* node_modules/*
dist/* dist/*
yarn.lock
yarn-error.log
visualizer.html visualizer.html
pnpm-lock.yaml
.idea .idea
auto-imports.d.ts auto-imports.d.ts
components.d.ts components.d.ts

View File

@ -1,5 +1,31 @@
# CHANGE LOG # CHANGE LOG
## 4.6.2
## Feats
- `AppMenu Extra` 已经稳定发布,现在你可以在菜单标题后面添加标记了,相关变更
- `extraIcon` 类型变为 `VNode`
- 新增 `show` 配置项,用于配置是否显示 `extra` 标记,如果默认为 `false`,则不会显示 `extra` 标记,如果未配置或者配置为 `true`,则会显示 `extra` 标记。意味着只要你配置了 `label` 则会显示标记
- 新增 `useBadge hooks` 方法,便捷操作菜单标记
- `update`:更新标记
- `show`:显示标记
- `hidden`:隐藏标记
- 新增 `i18nLabel` 配置项,当你希望该标签能够跟随模板语言切换动态切换的时候你可能用的上,该配置项优先级高于 `label`
> 该配置项仅在 `collapsed``false` 时生效。
- 更新 `axios` 版本至 `1.6.7`
- 移除 `matchMenuOption` 方法
- 调整 `vueRouterRegister` 方法的运行时机,现在会在 `router` 注册完成后再执行该方法
- 优化一些试例页面代码
- `RTable` 相关
- 新增 `Props` 组件,支持勾选配置一些表格的配置项,目前仅支持:边框、条纹
- 优化 `Size` 组件,取消没必要的响应式代理数据
- 取消自定义工具组件的 `Popover` 提示
- 更改 `config.ts` 文件名为 `shared.ts`,并且 `config` 配置导入导出方式更改为 `import { config } from '@/shared'`
- 更改 `README-ZH.md` 文件名为 `README.zh-CN.md`
## 4.6.2-beta1.2 ## 4.6.2-beta1.2
## Feats ## Feats

View File

@ -23,7 +23,7 @@
# Ray Template # Ray Template
English | [简体中文](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/README-ZH.md) English | [简体中文](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/README.zh-CN.md)
A `completely free`, `efficient`, `feature complete` and based on vite5. x & ts(x) & pinia & vue3. x and other latest technology in the background template. A `completely free`, `efficient`, `feature complete` and based on vite5. x & ts(x) & pinia & vue3. x and other latest technology in the background template.

View File

@ -1,7 +1,7 @@
{ {
"name": "ray-template", "name": "ray-template",
"private": false, "private": false,
"version": "4.6.2-beta1.2", "version": "4.6.2",
"type": "module", "type": "module",
"engines": { "engines": {
"node": "^18.0.0 || >=20.0.0", "node": "^18.0.0 || >=20.0.0",
@ -34,7 +34,7 @@
"dependencies": { "dependencies": {
"@vueuse/core": "^10.7.1", "@vueuse/core": "^10.7.1",
"awesome-qr": "2.1.5-rc.0", "awesome-qr": "2.1.5-rc.0",
"axios": "^1.6.5", "axios": "^1.6.7",
"clipboard": "^2.0.11", "clipboard": "^2.0.11",
"crypto-js": "^4.1.1", "crypto-js": "^4.1.1",
"currency.js": "^2.0.4", "currency.js": "^2.0.4",

8
pnpm-lock.yaml generated
View File

@ -12,8 +12,8 @@ dependencies:
specifier: 2.1.5-rc.0 specifier: 2.1.5-rc.0
version: 2.1.5-rc.0 version: 2.1.5-rc.0
axios: axios:
specifier: ^1.6.5 specifier: ^1.6.7
version: 1.6.5 version: 1.6.7
clipboard: clipboard:
specifier: ^2.0.11 specifier: ^2.0.11
version: 2.0.11 version: 2.0.11
@ -2810,8 +2810,8 @@ packages:
- supports-color - supports-color
dev: false dev: false
/axios@1.6.5: /axios@1.6.7:
resolution: {integrity: sha512-Ii012v05KEVuUoFWmMW/UQv9aRIc3ZwkWDcM+h5Il8izZCtRVpDUfwpoFf7eOtajT3QiGR4yDUx7lPqHJULgbg==} resolution: {integrity: sha512-/hDJGff6/c7u0hDkvkGxR/oy6CbCs8ziCsC7SqmhjfozqiJGc8Z11wrv9z9lYfY4K8l+H9TpjcMDX0xOZmx+RA==}
dependencies: dependencies:
follow-redirects: 1.15.5 follow-redirects: 1.15.5
form-data: 4.0.0 form-data: 4.0.0

View File

@ -81,13 +81,16 @@ export const APP_MENU_CONFIG: Readonly<AppMenuConfig> = {
* *
* key * key
* key 使 getStorage setStorage key * key 使 getStorage setStorage key
* APP_CATCH_KEY 使
*
* cache.ts
* *
* *
* *
* @example * @example
* export const APP_CATCH_KEY_PREFIX: = 'ray-template:' * export const APP_CATCH_KEY_PREFIX = 'ray-template:'
* *
* 'ray-template:signing' * 'ray-template:signing' // 会自动拼接为
*/ */
export const APP_CATCH_KEY_PREFIX = '' export const APP_CATCH_KEY_PREFIX = ''

View File

@ -16,14 +16,15 @@ import Size from './components/Size'
import Fullscreen from './components/Fullscreen' import Fullscreen from './components/Fullscreen'
import C from './components/C' import C from './components/C'
import Print from './components/Print' import Print from './components/Print'
import TablePropsSelect from './components/Props'
import props from './props' import props from './props'
import { call, renderNode, uuid } from '@/utils' import { call, renderNode, uuid } from '@/utils'
import config from './config' import { config } from './shared'
import type { DropdownOption, DataTableInst } from 'naive-ui' import type { DropdownOption, DataTableInst } from 'naive-ui'
import type { ComponentSize } from '@/types' import type { ComponentSize } from '@/types'
import type { C as CType } from './type' import type { C as CType, PropsComponentPopselectKeys } from './type'
export default defineComponent({ export default defineComponent({
name: 'RTable', name: 'RTable',
@ -54,6 +55,10 @@ export default defineComponent({
const privateReactive = reactive({ const privateReactive = reactive({
size: props.size, size: props.size,
}) })
const propsPopselectValue = ref({
striped: false,
bordered: false,
})
/** /**
* *
@ -139,6 +144,16 @@ export default defineComponent({
.map((curr) => (typeof curr === 'function' ? curr() : curr)) .map((curr) => (typeof curr === 'function' ? curr() : curr))
} }
const popselectChange = (value: PropsComponentPopselectKeys[]) => {
const keys = Object.keys(propsPopselectValue.value)
keys.forEach((key) => {
propsPopselectValue.value[key] = value.includes(
key as PropsComponentPopselectKeys,
)
})
}
/** /**
* *
* @param p props * @param p props
@ -152,6 +167,11 @@ export default defineComponent({
<Size {...p} onChangeSize={changeTableSize.bind(this)} /> <Size {...p} onChangeSize={changeTableSize.bind(this)} />
<Fullscreen /> <Fullscreen />
<C {...p} onUpdateColumn={updateTableColumn.bind(this)} /> <C {...p} onUpdateColumn={updateTableColumn.bind(this)} />
<TablePropsSelect
{...p}
onPopselectChange={popselectChange.bind(this)}
onInitialed={popselectChange.bind(this)}
/>
</> </>
) )
@ -192,59 +212,75 @@ export default defineComponent({
privateReactive, privateReactive,
tool, tool,
wrapperRef, wrapperRef,
propsPopselectValue,
} }
}, },
render() { render() {
const { tool } = this const {
$props,
$attrs,
wrapperBordered,
uuidWrapper,
privateReactive,
disabledContextMenu,
contextMenuReactive,
contextMenuOptions,
uuidTable,
title,
$slots,
propsPopselectValue,
} = this
const { tool, combineRowProps, contextMenuSelect } = this
return ( return (
<NCard <NCard
ref="wrapperRef" ref="wrapperRef"
bordered={this.wrapperBordered} bordered={wrapperBordered}
{...{ id: this.uuidWrapper }} {...{ id: uuidWrapper }}
> >
{{ {{
default: () => ( default: () => (
<> <>
<NDataTable <NDataTable
ref="rTableInst" ref="rTableInst"
{...{ id: this.uuidTable }} {...{ id: uuidTable }}
{...this.$props} {...$attrs}
{...this.$attrs} {...$props}
rowProps={this.combineRowProps.bind(this)} {...propsPopselectValue}
size={this.privateReactive.size} rowProps={combineRowProps.bind(this)}
size={privateReactive.size}
> >
{{ {{
...this.$slots, ...$slots,
}} }}
</NDataTable> </NDataTable>
{!this.disabledContextMenu ? ( {!disabledContextMenu ? (
<NDropdown <NDropdown
show={this.contextMenuReactive.showContextMenu} show={contextMenuReactive.showContextMenu}
placement="bottom-start" placement="bottom-start"
trigger="manual" trigger="manual"
x={this.contextMenuReactive.x} x={contextMenuReactive.x}
y={this.contextMenuReactive.y} y={contextMenuReactive.y}
options={this.contextMenuOptions} options={contextMenuOptions}
onClickoutside={() => onClickoutside={() =>
(this.contextMenuReactive.showContextMenu = false) (contextMenuReactive.showContextMenu = false)
} }
onSelect={this.contextMenuSelect.bind(this)} onSelect={contextMenuSelect.bind(this)}
/> />
) : null} ) : null}
</> </>
), ),
header: renderNode(this.title, { header: renderNode(title, {
defaultElement: <div style="display: none;"></div>, defaultElement: <div style="display: none;"></div>,
}), }),
'header-extra': () => ( 'header-extra': () => (
<NFlex align="center"> <NFlex align="center">
{/* eslint-disable @typescript-eslint/no-explicit-any */} {/* eslint-disable @typescript-eslint/no-explicit-any */}
{tool(this.$props as any)} {tool($props as any)}
</NFlex> </NFlex>
), ),
footer: () => this.$slots.tableFooter?.(), footer: () => $slots.tableFooter?.(),
action: () => this.$slots.tableAction?.(), action: () => $slots.tableAction?.(),
}} }}
</NCard> </NCard>
) )

View File

@ -21,7 +21,7 @@
import { NPopover, NFlex, NTree } from 'naive-ui' import { NPopover, NFlex, NTree } from 'naive-ui'
import { RIcon } from '@/components' import { RIcon } from '@/components'
import config from '../config' import { config } from '../shared'
import props from '../props' import props from '../props'
import { call } from '@/utils' import { call } from '@/utils'
@ -33,7 +33,7 @@ import type { MaybeArray } from '@/types'
type FixedClick = (type: 'left' | 'right', option: C, index: number) => void type FixedClick = (type: 'left' | 'right', option: C, index: number) => void
const renderSwitcherIcon = () => ( const renderSwitcherIcon = () => (
<RIcon name="draggable" size={config.tableIconSize} cursor="all-scroll" /> <RIcon name="draggable" size={14} cursor="all-scroll" />
) )
const RowIconRender = ({ const RowIconRender = ({

View File

@ -9,10 +9,9 @@
* @remark * @remark
*/ */
import { NPopover } from 'naive-ui'
import { RIcon } from '@/components' import { RIcon } from '@/components'
import config from '../config' import { config } from '../shared'
import { useFullscreen } from 'vue-hooks-plus' import { useFullscreen } from 'vue-hooks-plus'
import type { TableProvider } from '../type' import type { TableProvider } from '../type'
@ -29,7 +28,6 @@ export default defineComponent({
return { return {
toggleFullscreen, toggleFullscreen,
isFullscreen,
isEnabled, isEnabled,
} }
}, },
@ -37,27 +35,18 @@ export default defineComponent({
const { toggleFullscreen, isEnabled, $t } = this const { toggleFullscreen, isEnabled, $t } = this
return ( return (
<NPopover showArrow={false}> <RIcon
{{ name="fullscreen"
trigger: () => ( size={config.tableIconSize}
<RIcon cursor="pointer"
name="fullscreen" onClick={() => {
size={config.tableIconSize} if (!isEnabled) {
cursor="pointer" window.$message.warning($t('globalMessage.isEnabledFullscreen'))
onClick={() => { }
if (!isEnabled) {
window.$message.warning(
$t('globalMessage.isEnabledFullscreen'),
)
}
toggleFullscreen() toggleFullscreen()
}}
/>
),
default: () => (this.isFullscreen ? '取消全屏' : '全屏表格'),
}} }}
</NPopover> />
) )
}, },
}) })

View File

@ -9,10 +9,9 @@
* @remark * @remark
*/ */
import { NPopover } from 'naive-ui'
import { RIcon } from '@/components' import { RIcon } from '@/components'
import config from '../config' import { config } from '../shared'
import props from '../props' import props from '../props'
import { printDom } from '@/utils/dom' import { printDom } from '@/utils/dom'
@ -53,19 +52,12 @@ export default defineComponent({
}, },
render() { render() {
return ( return (
<NPopover showArrow={false}> <RIcon
{{ name="print"
trigger: () => ( size={config.tableIconSize}
<RIcon cursor="pointer"
name="print" onClick={this.printTableClick.bind(this)}
size={config.tableIconSize} />
cursor="pointer"
onClick={this.printTableClick.bind(this)}
/>
),
default: () => '打印表格',
}}
</NPopover>
) )
}, },
}) })

View File

@ -0,0 +1,104 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-10-04
*
* @workspace ray-template
*
* @remark
*/
import { NPopselect } from 'naive-ui'
import { RIcon } from '@/components'
import { call } from '@/utils'
import { config } from '../shared'
import props from '../props'
import type { MaybeArray } from '@/types'
import type { PropsComponentPopselectKeys } from '../type'
export default defineComponent({
name: 'TablePropsSelect',
props: {
...props,
onPopselectChange: {
type: [Function, Array] as PropType<
MaybeArray<(size: PropsComponentPopselectKeys[]) => void>
>,
default: null,
},
onInitialed: {
type: [Function, Array] as PropType<
MaybeArray<(size: PropsComponentPopselectKeys[]) => void>
>,
default: null,
},
},
setup(props) {
const popoverShow = ref(false)
const propsPopselectValue = ref<PropsComponentPopselectKeys[]>([])
const propsOptions = [
{
label: '斑马条纹',
value: 'striped',
},
{
label: '表格边框',
value: 'bordered',
},
]
const updatePopselectValue = (value: PropsComponentPopselectKeys[]) => {
const { onPopselectChange } = props
if (onPopselectChange) {
call(onPopselectChange, value)
}
}
const initialPopselectValue = () => {
const { striped, bordered, onInitialed } = props
if (striped) {
propsPopselectValue.value.push('striped')
}
if (bordered) {
propsPopselectValue.value.push('bordered')
}
if (onInitialed) {
call(onInitialed, propsPopselectValue.value)
}
}
initialPopselectValue()
return {
propsPopselectValue,
propsOptions,
popoverShow,
updatePopselectValue,
}
},
render() {
return (
<NPopselect
v-model:value={this.propsPopselectValue}
options={this.propsOptions}
trigger="click"
multiple
onUpdateValue={this.updatePopselectValue.bind(this)}
>
<RIcon
name="expanded"
size={config.tableIconSize}
cursor="pointer"
style="transform: rotate(270deg);"
/>
</NPopselect>
)
},
})

View File

@ -9,12 +9,12 @@
* @remark * @remark
*/ */
import { NPopover, NPopselect } from 'naive-ui' import { NPopselect } from 'naive-ui'
import { RIcon } from '@/components' import { RIcon } from '@/components'
import { call } from '@/utils' import { call } from '@/utils'
import props from '../props' import props from '../props'
import config from '../config' import { config } from '../shared'
import type { ComponentSize } from '@/types' import type { ComponentSize } from '@/types'
import type { MaybeArray } from '@/types' import type { MaybeArray } from '@/types'
@ -31,7 +31,6 @@ export default defineComponent({
...props, ...props,
}, },
setup(props) { setup(props) {
const popoverShow = ref(false)
const size = ref(props.size) const size = ref(props.size)
const sizeOptions = [ const sizeOptions = [
{ {
@ -59,7 +58,6 @@ export default defineComponent({
return { return {
size, size,
sizeOptions, sizeOptions,
popoverShow,
updatePopselectValue, updatePopselectValue,
} }
}, },
@ -71,18 +69,7 @@ export default defineComponent({
trigger="click" trigger="click"
onUpdateValue={this.updatePopselectValue.bind(this)} onUpdateValue={this.updatePopselectValue.bind(this)}
> >
<NPopover showArrow={false}> <RIcon name="adjustment" size={config.tableIconSize} cursor="pointer" />
{{
trigger: () => (
<RIcon
name="adjustment"
size={config.tableIconSize}
cursor="pointer"
/>
),
default: () => '密度',
}}
</NPopover>
</NPopselect> </NPopselect>
) )
}, },

View File

@ -11,14 +11,22 @@
import { dataTableProps } from 'naive-ui' import { dataTableProps } from 'naive-ui'
import type { PropType, VNode, VNodeChild } from 'vue' import type { PropType, VNode } from 'vue'
import type { MaybeArray } from '@/types' import type { MaybeArray } from '@/types'
import type { DropdownOption, DataTableColumn } from 'naive-ui' import type { DropdownOption, DataTableColumn } from 'naive-ui'
import type { DownloadTableOptions, PrintTableOptions } from './type' import type { DownloadCsvTableOptions, PrintTableOptions } from './type'
import type { Recordable } from '@/types' import type { Recordable } from '@/types'
const props = { const props = {
...dataTableProps, ...dataTableProps,
downloadCsvTableOptions: {
/**
*
*
*/
type: Object as PropType<DownloadCsvTableOptions>,
default: () => ({}),
},
title: { title: {
/** /**
* *
@ -61,14 +69,6 @@ const props = {
>, >,
default: null, default: null,
}, },
downloadTableOptions: {
/**
*
* excel
*/
type: Object as PropType<DownloadTableOptions>,
default: () => ({}),
},
wrapperBordered: { wrapperBordered: {
/** /**
* *
@ -86,16 +86,6 @@ const props = {
type: Object as PropType<PrintTableOptions>, type: Object as PropType<PrintTableOptions>,
default: () => ({}), default: () => ({}),
}, },
onDownloadSuccess: {
/** 导出表格成功回调 */
type: [Function, Array] as PropType<MaybeArray<() => void>>,
default: null,
},
onDownloadError: {
/** 导出表格失败回调 */
type: [Function, Array] as PropType<MaybeArray<() => void>>,
default: null,
},
onUpdateColumns: { onUpdateColumns: {
type: [Function, Array] as PropType< type: [Function, Array] as PropType<
MaybeArray<(arr: DataTableColumn[]) => void> MaybeArray<(arr: DataTableColumn[]) => void>

View File

@ -9,7 +9,7 @@
* @remark * @remark
*/ */
export default { export const config = {
tableIconSize: '18', tableIconSize: '18',
tableKey: Symbol('r-table'), tableKey: Symbol('r-table'),
} }

View File

@ -19,8 +19,9 @@ export type DropdownMixedOption =
| DropdownDividerOption | DropdownDividerOption
| DropdownRenderOption | DropdownRenderOption
export interface DownloadTableOptions { export interface DownloadCsvTableOptions {
fileName?: string fileName?: string
keepOriginalData?: boolean
} }
export interface PrintTableOptions extends PrintDomOptions {} export interface PrintTableOptions extends PrintDomOptions {}
@ -29,7 +30,6 @@ export interface TableProvider {
uuidWrapper: string uuidWrapper: string
uuidTable: string uuidTable: string
wrapperRef: Ref<HTMLElement | undefined> wrapperRef: Ref<HTMLElement | undefined>
tableRef: Ref<HTMLElement | undefined>
} }
export interface C extends DataTableBaseColumn { export interface C extends DataTableBaseColumn {
@ -44,6 +44,8 @@ export interface C extends DataTableBaseColumn {
export type OverridesTableColumn<T = Recordable> = C | DataTableColumn<T> export type OverridesTableColumn<T = Recordable> = C | DataTableColumn<T>
export interface TableInst extends TableProvider { export interface TableInst extends Omit<TableProvider, 'wrapperRef'> {
rTableInst: Omit<DataTableInst, 'clearFilter'> rTableInst: Omit<DataTableInst, 'clearFilter'>
} }
export type PropsComponentPopselectKeys = 'striped' | 'bordered'

View File

@ -1,21 +1,162 @@
import { useMenuGetters } from '@/store' import { useMenuGetters } from '@/store'
import { isValueType } from '@/utils'
import { createMenuExtra } from '@/store/modules/menu/helper'
export function useBadge() { import type { AppMenuOption } from '@/types'
import type { AppMenuExtraOptions } from '@/router/type'
export type BadgeKey = string | AppMenuOption
let cachePreNormal: AppMenuOption | undefined = void 0
// 渲染 extra
const renderExtra = (
option: AppMenuOption,
assignExtra: AppMenuExtraOptions,
) => {
const { show, ...params } = assignExtra
option.meta.extra = Object.assign({}, option.meta.extra, {
...params,
show: show,
})
option.extra = createMenuExtra(option)
cachePreNormal = option
}
/**
*
* @param path1 1
* @param path2 2
*
* @description
*
*
* @example
* equal('/a/', '/a') // true
* equal('/a', '/a') // true
*/
const equal = (path1: string, path2: string) => {
const path1End = path1.endsWith('/')
const path2End = path2.endsWith('/')
if (path1End && path2End) {
return path1.slice(0, -1) === path2.slice(0, -1)
}
if (!path1End && !path2End) {
return path1 === path2
}
return (
path1 === path2 ||
path1.slice(0, -1) === path2 ||
path1 === path2.slice(0, -1)
)
}
// 递归查找匹配的菜单项,缓存上一次的匹配项
const deep = (
options: AppMenuOption[],
target: string,
fn: string,
assignExtra: AppMenuExtraOptions,
) => {
if (cachePreNormal && equal(cachePreNormal.fullPath, target)) {
renderExtra(cachePreNormal, assignExtra)
return cachePreNormal
}
for (const curr of options) {
if (equal(curr.fullPath, target)) {
renderExtra(curr, assignExtra)
cachePreNormal = curr
return curr
}
if (curr.children?.length) {
deep(curr.children, target, fn, assignExtra)
continue
}
}
}
// 匹配菜单项
const normalOption = (
target: BadgeKey,
fn: string,
assignExtra: AppMenuExtraOptions,
) => {
const { getMenuOptions } = useMenuGetters() const { getMenuOptions } = useMenuGetters()
const hidden = () => {} if (typeof target === 'string') {
deep(getMenuOptions.value, target, fn, assignExtra)
} else if (isValueType<'object'>(target, 'Object')) {
const { fullPath } = target
const show = () => {} deep(getMenuOptions.value, fullPath, fn, assignExtra)
} else {
console.warn(`[useBadge ${fn}]: target expect string or object.`)
}
}
const toggle = () => {} export function useBadge() {
/**
*
* @param target key AppMenuOption)
*
* @example
* const { hidden } = useBadge()
*
* hidden('your key')
* hidden({ ...AppMenuOption })
*/
const hidden = (target: BadgeKey) => {
normalOption(target, 'hidden', {
show: false,
})
}
const updateLabel = () => {} /**
*
* @param target key AppMenuOption)
*
* @example
* const { show } = useBadge()
*
* show('your key')
* show({ ...AppMenuOption })
*/
const show = (target: BadgeKey) => {
normalOption(target, 'show', {
show: true,
})
}
/**
*
* @param target key AppMenuOption)
* @param extraOption
*
* @example
* const { update } = useBadge()
*
* update('your key', { ...AppMenuExtraOptions })
* update({ ...AppMenuOption }, { ...AppMenuExtraOptions })
*/
const update = (target: BadgeKey, extraOption: AppMenuExtraOptions) => {
normalOption(target, 'update', extraOption)
}
return { return {
hidden, hidden,
show, show,
toggle, update,
updateLabel,
} }
} }

View File

@ -29,8 +29,8 @@ const createVueRouter = async () => {
export const setupRouter = async (app: App<Element>) => { export const setupRouter = async (app: App<Element>) => {
router = await createVueRouter() router = await createVueRouter()
vueRouterRegister(router)
app.use(router) app.use(router)
// 等待 router 挂载后,初始化 useRouter 方法 // 等待 router 挂载后,初始化 useRouter 方法
useVueRouter() useVueRouter()
vueRouterRegister(router)
} }

View File

@ -11,7 +11,9 @@ const cacheDemo: AppRouteRecordRaw = {
i18nKey: t('menu.CacheDemo'), i18nKey: t('menu.CacheDemo'),
icon: 'other', icon: 'other',
order: 3, order: 3,
extra: 'new', extra: {
label: 'new',
},
}, },
} }

View File

@ -11,6 +11,9 @@ const axios: AppRouteRecordRaw = {
i18nKey: t('menu.TemplateHooks'), i18nKey: t('menu.TemplateHooks'),
icon: 'other', icon: 'other',
order: 1, order: 1,
extra: {
label: 'new',
},
}, },
} }

View File

@ -11,9 +11,11 @@ export type Component<T = any> =
| (() => Promise<T>) | (() => Promise<T>)
export interface AppMenuExtraOptions { export interface AppMenuExtraOptions {
extraLabel?: string | number label?: string | number
extraIcon?: string | VNode icon?: VNode
extraType?: TagProps['type'] type?: TagProps['type']
show?: boolean
i18nLabel?: string
} }
export interface AppRouteMeta { export interface AppRouteMeta {
@ -28,7 +30,7 @@ export interface AppRouteMeta {
keepAlive?: boolean keepAlive?: boolean
sameLevel?: boolean sameLevel?: boolean
env?: string | string[] env?: string | string[]
extra?: string | AppMenuExtraOptions extra?: AppMenuExtraOptions
} }
// @ts-ignore // @ts-ignore

View File

@ -11,17 +11,14 @@
/** 本方法感谢 <https://yunkuangao.me/> 的支持 */ /** 本方法感谢 <https://yunkuangao.me/> 的支持 */
import { APP_MENU_CONFIG, APP_CATCH_KEY } from '@/app-config'
import { RIcon } from '@/components' import { RIcon } from '@/components'
import { APP_MENU_CONFIG, APP_CATCH_KEY } from '@/app-config'
import { getStorage, isValueType } from '@/utils' import { getStorage, isValueType } from '@/utils'
import { useAppRoot } from '@/hooks' import { useAppRoot, useI18n } from '@/hooks'
import { NTag } from 'naive-ui' import { NTag } from 'naive-ui'
import type { import type { AppMenuOption, AppMenuKey } from '@/types/modules/app'
AppMenuOption,
MenuTagOptions,
AppMenuKey,
} from '@/types/modules/app'
import type { TagProps } from 'naive-ui' import type { TagProps } from 'naive-ui'
/** /**
@ -106,28 +103,6 @@ export const parseAndFindMatchingNodes = (
return temp return temp
} }
/**
*
* @param item menu options
* @param key current menu key
* @param menuTagOptions menu tag options
*
* @remark
*/
export const matchMenuOption = (
item: AppMenuOption,
key: AppMenuKey,
menuTagOptions: MenuTagOptions[],
) => {
if (item.path !== key) {
const tag = menuTagOptions.find((curr) => curr.path === item.path)
if (!tag) {
menuTagOptions.push(item)
}
}
}
/** /**
* *
* @param option menu option * @param option menu option
@ -145,6 +120,18 @@ export const updateDocumentTitle = (option: AppMenuOption) => {
document.title = breadcrumbLabel + ' - ' + spliceTitle document.title = breadcrumbLabel + ' - ' + spliceTitle
} }
/**
*
* @param option menu option
*
* @returns
*
* @description
* AppMenuOption AppMenuOption
*
* @example
* createMenuIcon({ ...AppMenuOption })
*/
export const createMenuIcon = (option: AppMenuOption) => { export const createMenuIcon = (option: AppMenuOption) => {
const { const {
meta: { icon }, meta: { icon },
@ -171,6 +158,18 @@ export const createMenuIcon = (option: AppMenuOption) => {
return () => _icon return () => _icon
} }
/**
*
* @param option menu option
*
* @returns
*
* @description
* AppMenuOption AppMenuOption
*
* @example
* createMenuExtra({ ...AppMenuOption })
*/
export const createMenuExtra = (option: AppMenuOption) => { export const createMenuExtra = (option: AppMenuOption) => {
const { const {
meta: { extra }, meta: { extra },
@ -180,6 +179,14 @@ export const createMenuExtra = (option: AppMenuOption) => {
return return
} }
const { show } = extra
if (show === false) {
return
}
const { t } = useI18n()
const { label, icon, type, i18nLabel } = extra
const tagProps: TagProps = { const tagProps: TagProps = {
type: 'primary', type: 'primary',
size: 'small', size: 'small',
@ -188,32 +195,58 @@ export const createMenuExtra = (option: AppMenuOption) => {
strong: true, strong: true,
} }
if (isValueType<object>(extra, 'Object')) { // 渲染文案优先级i18nLabel > label
const { extraLabel, extraIcon, extraType } = extra const renderLabel = () => {
if (i18nLabel) {
return t(i18nLabel)
}
if (label) {
return label
}
return null
}
// 如果没有文案,但是设置了 icon则将 padding-right 设置为 0
const style = () => {
if ((icon && (i18nLabel || label)) || (!icon && (i18nLabel || label))) {
return null
}
return {
'padding-right': '0',
}
}
if (isValueType<object>(extra, 'Object')) {
return () => { return () => {
return h( return h(
NTag, NTag,
{ {
...tagProps, ...tagProps,
type: extraType || 'primary', type: type || 'primary',
style: style(),
}, },
{ {
default: () => extraLabel, default: () => renderLabel(),
icon: () => extraIcon, icon: () => icon,
}, },
) )
} }
} }
return () => {
return h(NTag, tagProps, {
default: () => extra,
})
}
} }
/** 获取缓存的 menu key, 如果未获取到则使用 getRootPath 当作默认激活路由菜单 */ /**
*
* @returns key
*
* @description
* key
*
* @example
* getCatchMenuKey() // '/dashboard'
*/
export const getCatchMenuKey = () => { export const getCatchMenuKey = () => {
const { getRootPath } = useAppRoot() const { getRootPath } = useAppRoot()
const cacheMenuKey = getStorage<AppMenuKey>( const cacheMenuKey = getStorage<AppMenuKey>(

View File

@ -20,6 +20,7 @@
.r-menu--app .n-menu-item-content .n-menu-item-content-header { .r-menu--app .n-menu-item-content .n-menu-item-content-header {
display: flex; display: flex;
gap: 0 6px; gap: 0 6px;
align-items: center;
} }
.r-menu--app { .r-menu--app {

View File

@ -13,6 +13,7 @@ export interface AppMenuOption extends AppRouteRecordRaw {
meta: AppRouteMeta meta: AppRouteMeta
breadcrumbLabel?: string breadcrumbLabel?: string
fullPath: string fullPath: string
extra?: string | (() => VNode)
} }
export interface MenuTagOptions extends AppMenuOption { export interface MenuTagOptions extends AppMenuOption {

View File

@ -20,35 +20,7 @@ import type { Person } from '@/api/demo/mock/person'
const MockDemo = defineComponent({ const MockDemo = defineComponent({
name: 'MockDemo', name: 'MockDemo',
setup() { setup() {
const paginationRef = reactive({ const columns = [
page: 1,
pageSize: 10,
itemCount: 0,
pageSizes: [10, 20, 30, 40, 50],
showSizePicker: true,
onUpdatePage: (page: number) => {
paginationRef.page = page
getPerson()
},
onUpdatePageSize: (pageSize: number) => {
paginationRef.pageSize = pageSize
paginationRef.page = 1
getPerson()
},
})
const {
data: personData,
loading: personLoading,
run: personFetchRun,
} = useHookPlusRequest(getPersonList, {
manual: true,
onSuccess: (data) => {
console.log(data)
},
})
const columns = ref([
{ {
title: 'id', title: 'id',
key: 'id', key: 'id',
@ -110,10 +82,46 @@ const MockDemo = defineComponent({
) )
}, },
}, },
]) ]
const condition = reactive({ const condition = reactive({
email: null, email: null,
}) })
const paginationRef = reactive({
page: 1,
pageSize: 10,
itemCount: 0,
pageSizes: [10, 20, 30, 40, 50],
showSizePicker: true,
onUpdatePage: (page: number) => {
paginationRef.page = page
getPerson()
},
onUpdatePageSize: (pageSize: number) => {
paginationRef.pageSize = pageSize
paginationRef.page = 1
getPerson()
},
})
const {
data: personData,
loading: personLoading,
run: personFetchRun,
} = useHookPlusRequest(getPersonList, {
defaultParams: [
{
page: paginationRef.page,
pageSize: paginationRef.pageSize,
email: condition.email,
},
],
onSuccess: (res) => {
const { total } = res
paginationRef.itemCount = total
},
})
const getPerson = () => { const getPerson = () => {
const { pageSize, page } = paginationRef const { pageSize, page } = paginationRef
@ -126,16 +134,6 @@ const MockDemo = defineComponent({
}) })
} }
watchEffect(() => {
if (personData.value) {
paginationRef.itemCount = personData.value.total
}
})
onBeforeMount(() => {
getPerson()
})
return { return {
personData, personData,
personLoading, personLoading,
@ -180,7 +178,7 @@ const MockDemo = defineComponent({
</RCollapseGrid> </RCollapseGrid>
</NForm> </NForm>
<RTable <RTable
title="分页表格" title="Mock数据表格"
data={this.personData?.data} data={this.personData?.data}
loading={this.personLoading} loading={this.personLoading}
v-model:columns={this.columns} v-model:columns={this.columns}

View File

@ -10,6 +10,7 @@
*/ */
import { NFlex, NCard, NButton, NInput } from 'naive-ui' import { NFlex, NCard, NButton, NInput } from 'naive-ui'
import { RIcon } from '@/components'
import { import {
useAppNavigation, useAppNavigation,
@ -17,6 +18,7 @@ import {
useSpinning, useSpinning,
useWatermark, useWatermark,
useTheme, useTheme,
useBadge,
} from '@/hooks' } from '@/hooks'
import { getVariableToRefs } from '@/global-variable' import { getVariableToRefs } from '@/global-variable'
import { useSettingGetters } from '@/store' import { useSettingGetters } from '@/store'
@ -27,6 +29,7 @@ export default defineComponent({
const currentMenuOption = ref('') const currentMenuOption = ref('')
const maximizeRef = getVariableToRefs('layoutContentMaximize') const maximizeRef = getVariableToRefs('layoutContentMaximize')
const watermark = ref(useSettingGetters().getWatermarkConfig.value.content) const watermark = ref(useSettingGetters().getWatermarkConfig.value.content)
const badgeValue = ref('new')
const { navigationTo } = useAppNavigation() const { navigationTo } = useAppNavigation()
const { maximize, isLayoutContentMaximized } = useMaximize() const { maximize, isLayoutContentMaximized } = useMaximize()
@ -39,6 +42,11 @@ export default defineComponent({
} = useWatermark() } = useWatermark()
const { changeDarkTheme, changeLightTheme, toggleTheme, getAppTheme } = const { changeDarkTheme, changeLightTheme, toggleTheme, getAppTheme } =
useTheme() useTheme()
const {
hidden: badgeHidden,
show: badgeShow,
update: badgeUpdateLabel,
} = useBadge()
return { return {
navigationTo, navigationTo,
@ -58,6 +66,10 @@ export default defineComponent({
toggleTheme, toggleTheme,
getAppTheme, getAppTheme,
isLayoutContentMaximized, isLayoutContentMaximized,
badgeHidden,
badgeShow,
badgeUpdateLabel,
badgeValue,
} }
}, },
render() { render() {
@ -76,6 +88,9 @@ export default defineComponent({
toggleTheme, toggleTheme,
getAppTheme, getAppTheme,
isLayoutContentMaximized, isLayoutContentMaximized,
badgeHidden,
badgeShow,
badgeUpdateLabel,
} = this } = this
return ( return (
@ -86,6 +101,48 @@ export default defineComponent({
</h3> </h3>
</NCard> </NCard>
<NCard title="useBadge 菜单标记">
<NFlex vertical>
<NInput v-model:value={this.badgeValue} />
<NFlex>
<NButton onClick={() => badgeHidden('/template-hooks')}>
</NButton>
<NButton onClick={() => badgeShow('/template-hooks')}>
</NButton>
<NButton
onClick={() =>
badgeUpdateLabel('/template-hooks', {
label: this.badgeValue,
})
}
>
</NButton>
<NButton
onClick={() => {
badgeUpdateLabel('/template-hooks', {
label: this.badgeValue,
icon: <RIcon name="error" size="18" />,
})
}}
>
</NButton>
<NButton
onClick={() => {
badgeUpdateLabel('/template-hooks', {
label: this.badgeValue,
icon: void 0,
})
}}
>
</NButton>
</NFlex>
</NFlex>
</NCard>
<NCard title="useTheme 主题"> <NCard title="useTheme 主题">
<NFlex vertical> <NFlex vertical>
<h3>getAppTheme : {getAppTheme().themeLabel}</h3> <h3>getAppTheme : {getAppTheme().themeLabel}</h3>

View File

@ -104,7 +104,7 @@ function onlyBuildOptions(mode: string) {
name: 'axios', name: 'axios',
global: 'axios', global: 'axios',
resolve: resolve:
'https://cdnjs.cloudflare.com/ajax/libs/axios/1.6.5/axios.min.js', 'https://cdnjs.cloudflare.com/ajax/libs/axios/1.6.7/axios.min.js',
}, },
], ],
}), }),