version: v4.7.0

This commit is contained in:
XiaoDaiGua-Ray 2024-03-09 23:39:59 +08:00
parent e20dbb4cd2
commit 70eee28aa8
128 changed files with 1144 additions and 960 deletions

View File

@ -25,6 +25,7 @@
"domtoimage",
"EDITMSG",
"iife",
"linebreak",
"macarons",
"menutag",
"ndata",
@ -32,6 +33,7 @@
"Popselect",
"precommit",
"siderbar",
"stylelint",
"WUJIE",
"zlevel"
]

View File

@ -1,5 +1,48 @@
# CHANGE LOG
## 4.7.0
做了一些核心依赖的升级操作。
并且规范了整个模板的包命名,这个一直算是遗留问题,有些包名不够语意化与有点混乱,现在终于统一了。
## Feats
- `useDomToImage` 相关
- 优化 `ts` 类型提示
- `usePrint` 相关
- 现在会强制剔除 `printable` 配置项
- 移除 `rollup-plugin-visualizer` 体积分析插件,使用 `vite-bundle-analyzer` 替换
```sh
# 执行
pnpm report
# 等待构建后,会自动打开浏览器。
```
- 移除 `report` 模式的 `eslint` 检查
- `axios` 相关
- `BeforeFetchFunction` 类型更名为 `FetchFunction`
- `AppRawRequestConfig` 类型新增 `__CANCELER_TAG_RAY_TEMPLATE__` 标记,用于标记是否需要可以被取消
- 优化 `ts` 类型标注
- 将所有 `type.ts` 包重命名为 `types.ts` 符合语义
- 更新 `vueuse` 版本至 `10.9.0`
- 更新 `vite` 版本至 `5.1.5`
- 更新 `vue` 版本至 `3.4.21`
- 将所有 `helper.ts, helper file` 统一更改为 `utils.ts`, `utils file` 方式管理
- 重构 `app/prefixCacheKey` 方法,现在支持自定义前缀
- 优化 `GlobalSearch` 搜索待选项样式
- `__ray-template` 包现在只会在 `__DEV__` 环境下才会做检查
- 新增 `ellipsis` 指令,并且补充所有自定义指令的注释
- `router` 包相关
- 修改 `router` 注册形式,改为同步注册
- 修改 `routes` 包导出形式,改为导出一个数组
## Fixes
- 修复 `useVueRouter` 方法 `HMR` 时可能会报错的问题
## 4.6.4
稳定了 `4.6.4` 版本。
@ -208,7 +251,7 @@ remove('your key', 'all')
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const cacheDemo: AppRouteRecordRaw = {
// ...your route config,

View File

@ -1,7 +1,7 @@
{
"name": "ray-template",
"private": false,
"version": "4.6.4",
"version": "4.7.0",
"type": "module",
"engines": {
"node": "^18.0.0 || >=20.0.0",
@ -32,7 +32,7 @@
]
},
"dependencies": {
"@vueuse/core": "^10.7.1",
"@vueuse/core": "^10.9.0",
"awesome-qr": "2.1.5-rc.0",
"axios": "^1.6.7",
"clipboard": "^2.0.11",
@ -48,7 +48,7 @@
"pinia": "^2.1.7",
"pinia-plugin-persistedstate": "^3.2.0",
"print-js": "^1.6.0",
"vue": "^3.4.20",
"vue": "^3.4.21",
"vue-demi": "0.14.6",
"vue-hooks-plus": "1.8.8",
"vue-i18n": "^9.9.0",
@ -87,13 +87,13 @@
"postcss": "^8.4.31",
"postcss-px-to-viewport-8-plugin": "1.2.3",
"prettier": "^3.2.5",
"rollup-plugin-visualizer": "^5.12.0",
"sass": "1.71.1",
"svg-sprite-loader": "^6.0.11",
"typescript": "^5.2.2",
"unplugin-auto-import": "^0.17.5",
"unplugin-vue-components": "^0.26.0",
"vite": "^5.1.4",
"vite": "^5.1.5",
"vite-bundle-analyzer": "0.8.1",
"vite-plugin-cdn2": "0.15.4",
"vite-plugin-compression": "^0.5.1",
"vite-plugin-ejs": "^1.7.0",

844
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@ -4,3 +4,11 @@
- validAppRootPath: 检查模板 `appRootPath` 是否配置正确
- validLocal: 检查模板 `localConfig` 是否配置正确
## 拓展
当你需要在做一些定制化操作的时候,可以尝试在这个包里做一些事情。
租后在 `main.ts` 中导入并且调用即可。
> 出于一些考虑,并没有做自动化导入调用,所以需要自己手动来。(好吧,其实就是我懒--

View File

@ -11,6 +11,10 @@ import { useVueRouter } from '@/hooks'
* getRoutes
*/
export const validAppRootPath = async () => {
if (!__DEV__) {
return
}
const { getAppRootRoute } = useSettingGetters()
const {
router: { getRoutes },

View File

@ -92,6 +92,10 @@ const validDefaultDayjsLocal = () => {
* localConfig
*/
export const validLocal = async () => {
if (!__DEV__) {
return
}
validSystemDefaultLocal()
validSystemFallbackLocale()
validDayjsLocalMap()

View File

@ -48,7 +48,7 @@ export const getWeather = (city: string) => {
})
}
export const getTypicode = () => {
export const getTypicCode = () => {
return request<JSONPlaceholder>({
url: 'https://jsonplaceholder.typicode.com/todos/1',
method: 'get',

View File

@ -29,7 +29,7 @@ import {
NModalProvider,
} from 'naive-ui'
import { naiveLocales } from '@/locales/helper'
import { getNaiveLocales } from '@/locales/utils'
import { useSettingGetters } from '@/store'
export default defineComponent({
@ -41,7 +41,7 @@ export default defineComponent({
const localePackage = computed(() => {
const key = getLocaleLanguage.value
return naiveLocales(key)
return getNaiveLocales(key)
})
/**

View File

@ -17,7 +17,7 @@
* beforeRouteUpdate -> cancelAllRequest -> routeUpdate
*/
import { axiosCanceler } from '@/axios/helper/interceptor'
import { axiosCanceler } from '@/axios/utils/interceptor'
const AppRequestCancelerProvider = defineComponent({
name: 'AppRequestCancelerProvider',

View File

@ -5,7 +5,7 @@
## 工具函数
- BeforeFetchFunction
- FetchFunction
- FetchErrorFunction
> 两个工具函数方便类型推导。
@ -28,13 +28,14 @@ import { appendRequestHeaders } from '@/axios/helper/axiosCopilot'
import type {
RequestInterceptorConfig,
BeforeFetchFunction,
FetchFunction,
FetchErrorFunction,
} from '@/axios/type'
} from '@/axios/types'
const injectRequestHeaderOfEnv: BeforeFetchFunction<
RequestInterceptorConfig
> = (ins, mode) => {
const injectRequestHeaderOfEnv: FetchFunction<RequestInterceptorConfig> = (
ins,
mode,
) => {
if (mode === 'development') {
appendRequestHeaders(ins, [
{

View File

@ -25,7 +25,7 @@ import useHookPlusRequest from 'vue-hooks-plus/es/useRequest'
import request from '@/axios/instance'
import type { UseRequestOptions } from 'vue-hooks-plus/es/useRequest/types'
import type { AppRawRequestConfig } from '@/axios/type'
import type { AppRawRequestConfig } from '@/axios/types'
/**
*
@ -48,11 +48,11 @@ function useRequest<
HookPlusParams extends unknown[] = unknown[],
HookPlusPlugin = unknown,
>(
fetchOption: AppRawRequestConfig<Response>,
fetchOptions: AppRawRequestConfig<Response>,
option?: UseRequestOptions<Response, HookPlusParams, HookPlusPlugin>,
) {
const fc = () => {
const cb = request<Response>(fetchOption)
const cb = request<Response>(fetchOptions)
return cb
}

View File

@ -9,7 +9,7 @@
* @remark
*/
import { useAxiosInterceptor } from '@/axios/helper/interceptor'
import { useAxiosInterceptor } from '@/axios/utils/interceptor'
import implement from './provider'
const { setImplement } = useAxiosInterceptor()

View File

@ -20,16 +20,16 @@
* injectRequestCanceler requestErrorCanceler axios request interceptor
*/
import { axiosCanceler } from '@/axios/helper/interceptor'
import { appendRequestHeaders } from '@/axios/helper/axiosCopilot'
import { axiosCanceler } from '@/axios/utils/interceptor'
import { appendRequestHeaders } from '@/axios/utils/axiosCopilot'
import { APP_CATCH_KEY } from '@/app-config'
import { getStorage } from '@/utils'
import type {
RequestInterceptorConfig,
BeforeFetchFunction,
FetchFunction,
FetchErrorFunction,
} from '@/axios/type'
} from '@/axios/types'
import type { Recordable } from '@/types'
/**
@ -54,10 +54,7 @@ const requestHeaderToken = (ins: RequestInterceptorConfig, mode: string) => {
}
/** 注入请求头信息 */
const injectRequestHeaders: BeforeFetchFunction<RequestInterceptorConfig> = (
ins,
mode,
) => {
const injectRequestHeaders: FetchFunction = (ins, mode) => {
appendRequestHeaders(ins, [
requestHeaderToken(ins, mode),
{
@ -72,12 +69,10 @@ const injectRequestHeaders: BeforeFetchFunction<RequestInterceptorConfig> = (
* @param ins
* @param mode
*
*
* @description
*
*/
const injectRequestCanceler: BeforeFetchFunction<RequestInterceptorConfig> = (
ins,
mode,
) => {
const injectRequestCanceler: FetchFunction = (ins, mode) => {
axiosCanceler.removePendingRequest(ins) // 检查是否存在重复请求, 若存在则取消已发的请求
axiosCanceler.addPendingRequest(ins) // 把当前的请求信息添加到 pendingRequest 表中
}
@ -87,10 +82,11 @@ const injectRequestCanceler: BeforeFetchFunction<RequestInterceptorConfig> = (
* @param error
* @param mode
*
*
* @description
*
*/
const requestErrorCanceler: FetchErrorFunction<Recordable> = (error, mode) => {
axiosCanceler.removePendingRequest(error)
const requestErrorCanceler: FetchErrorFunction = (error, mode) => {
axiosCanceler.removePendingRequest(error) // 移除请求拦截器
}
/**

View File

@ -9,7 +9,7 @@
* @remark
*/
import { useAxiosInterceptor } from '@/axios/helper/interceptor'
import { useAxiosInterceptor } from '@/axios/utils/interceptor'
import implement from './provider'
const { setImplement } = useAxiosInterceptor()

View File

@ -20,13 +20,13 @@
* injectResponseCanceler responseErrorCanceler axios response interceptor
*/
import { axiosCanceler } from '@/axios/helper/interceptor'
import { axiosCanceler } from '@/axios/utils/interceptor'
import type {
ResponseInterceptorConfig,
BeforeFetchFunction,
FetchFunction,
FetchErrorFunction,
} from '@/axios/type'
} from '@/axios/types'
import type { Recordable } from '@/types'
/**
@ -34,13 +34,11 @@ import type { Recordable } from '@/types'
* @param ins
* @param mode
*
*
* @description
*
*/
const injectResponseCanceler: BeforeFetchFunction<ResponseInterceptorConfig> = (
ins,
mode,
) => {
axiosCanceler.removePendingRequest(ins.config)
const injectResponseCanceler: FetchFunction = (ins, mode) => {
axiosCanceler.removePendingRequest(ins)
}
/**
@ -48,10 +46,11 @@ const injectResponseCanceler: BeforeFetchFunction<ResponseInterceptorConfig> = (
* @param error
* @param mode
*
*
* @description
*
*/
const responseErrorCanceler: FetchErrorFunction<Recordable> = (error, mode) => {
axiosCanceler.removePendingRequest(error.config)
const responseErrorCanceler: FetchErrorFunction = (error, mode) => {
axiosCanceler.removePendingRequest(error)
}
/**

View File

@ -18,7 +18,7 @@
import axios from 'axios'
import { AXIOS_CONFIG } from '@/app-config'
import { useAxiosInterceptor } from '@/axios/helper/interceptor'
import { useAxiosInterceptor } from '@/axios/utils/interceptor'
import {
setupResponseInterceptor,
setupResponseErrorInterceptor,
@ -28,7 +28,7 @@ import {
setupRequestErrorInterceptor,
} from '@/axios/inject/request'
import type { AxiosInstanceExpand } from './type'
import type { AxiosInstanceExpand } from './types'
const server: AxiosInstanceExpand = axios.create(AXIOS_CONFIG)
const { createAxiosInstance, beforeFetch, fetchError } = useAxiosInterceptor()

View File

@ -6,6 +6,7 @@ import type {
AxiosDefaults,
Axios,
AxiosResponse,
AxiosError,
} from 'axios'
import type { AnyFC } from '@/types'
@ -28,8 +29,13 @@ export interface CancelConfig {
export interface AppRawRequestConfig<T = any> extends AxiosRequestConfig<T> {
cancelConfig?: CancelConfig
__CANCELER_TAG_RAY_TEMPLATE__?: '__CANCELER_TAG_RAY_TEMPLATE__'
}
export interface CancelerParams<T = any, D = any>
extends AppRawRequestConfig<T>,
AxiosError<T, D> {}
export interface AxiosInstanceExpand extends Axios {
<T = any, D = any>(config: AppRawRequestConfig<D>): Promise<T>
<T = any, D = any>(url: string, config?: AppRawRequestConfig<D>): Promise<T>
@ -104,14 +110,15 @@ export interface ErrorImplementQueue {
implementResponseInterceptorErrorArray: AnyFC[]
}
export type BeforeFetchFunction<
T = RequestInterceptorConfig | ResponseInterceptorConfig,
> = <K extends T>(ins: K, mode: string) => void
export type FetchFunction = <T = any, K = any>(
ins: RequestInterceptorConfig<T> & ResponseInterceptorConfig<T, K>,
mode: string,
) => void
export type FetchType = 'ok' | 'error'
export type FetchErrorFunction<T = any> = <K extends T>(
error: K,
export type FetchErrorFunction<T = any, D = any> = (
error: AxiosError<T, D>,
mode: string,
) => void
@ -120,7 +127,7 @@ export interface AxiosFetchInstance {
responseInstance: ResponseInterceptorConfig | null
}
export interface AxiosFetchError<T = unknown> {
export interface AxiosFetchError<T = any> {
requestError: T | null
responseError: T | null
}

View File

@ -16,7 +16,7 @@
*
*/
import type { AppRawRequestConfig } from '@/axios/type'
import type { AppRawRequestConfig, CancelerParams } from '@/axios/types'
export default class RequestCanceler {
private pendingRequest: Map<string, AbortController>
@ -26,7 +26,7 @@ export default class RequestCanceler {
}
/** 是否需要加入取消请求表中 */
private isAppending(config: AppRawRequestConfig) {
private isAppending(config: AppRawRequestConfig | CancelerParams) {
return config.cancelConfig?.cancel ?? true
}
@ -37,7 +37,7 @@ export default class RequestCanceler {
*
* @remark config request key
*/
private generateRequestKey(config: AppRawRequestConfig) {
private generateRequestKey(config: AppRawRequestConfig | CancelerParams) {
const { method, url } = config
return [
@ -54,8 +54,10 @@ export default class RequestCanceler {
*
* @remark signal ,
*/
addPendingRequest(config: AppRawRequestConfig) {
addPendingRequest(config: AppRawRequestConfig | CancelerParams) {
if (this.isAppending(config)) {
config.__CANCELER_TAG_RAY_TEMPLATE__ = '__CANCELER_TAG_RAY_TEMPLATE__'
const requestKey = this.generateRequestKey(config)
if (!this.pendingRequest.has(requestKey)) {
@ -77,7 +79,7 @@ export default class RequestCanceler {
*
* @remark , map generateRequestKey value
*/
removePendingRequest(config: AppRawRequestConfig) {
removePendingRequest(config: AppRawRequestConfig | CancelerParams) {
const requestKey = this.generateRequestKey(config)
if (this.pendingRequest.has(requestKey)) {

View File

@ -12,7 +12,7 @@
/** axios 拦截器工具 */
import type { RawAxiosRequestHeaders, AxiosRequestConfig } from 'axios'
import type { RequestHeaderOptions } from '../type'
import type { RequestHeaderOptions } from '../types'
/**
*

View File

@ -20,7 +20,7 @@
* 使,
*/
import RequestCanceler from '@/axios/helper/RequestCanceler'
import RequestCanceler from '@/axios/utils/RequestCanceler'
import { getAppEnvironment } from '@/utils'
import type {
@ -31,8 +31,9 @@ import type {
FetchType,
AxiosFetchInstance,
AxiosFetchError,
} from '@/axios/type'
} from '@/axios/types'
import type { AnyFC } from '@/types'
import type { AxiosError } from 'axios'
/** 当前请求的实例 */
const axiosFetchInstance: AxiosFetchInstance = {
@ -40,7 +41,7 @@ const axiosFetchInstance: AxiosFetchInstance = {
responseInstance: null,
}
/** 请求失败返回值 */
const axiosFetchError: AxiosFetchError = {
const axiosFetchError: AxiosFetchError<AxiosError<unknown, unknown>> = {
requestError: null,
responseError: null,
}
@ -94,7 +95,7 @@ export const useAxiosInterceptor = () => {
/** 队列执行器 */
const implementer = (funcs: AnyFC[], ...args: any[]) => {
if (Array.isArray(funcs)) {
funcs?.forEach((curr) => {
funcs.forEach((curr) => {
if (typeof curr === 'function') {
curr(...args)
}
@ -123,7 +124,7 @@ export const useAxiosInterceptor = () => {
/** 请求、响应错误时执行队列中所有方法 */
const fetchError = (
key: keyof AxiosFetchError,
error: unknown,
error: AxiosError<unknown, unknown>,
errorImplementKey: keyof ErrorImplementQueue,
) => {
axiosFetchError[key] = error
@ -143,3 +144,5 @@ export const useAxiosInterceptor = () => {
fetchError,
}
}
export type UseAxiosInterceptor = ReturnType<typeof useAxiosInterceptor>

View File

@ -3,7 +3,7 @@ import chartProps from './src/props'
import type { ExtractPublicPropTypes } from 'vue'
import type * as RChartType from './src/type'
import type * as RChartType from './src/types'
export type ChartProps = ExtractPublicPropTypes<typeof chartProps>
export type { RChartType }

View File

@ -26,7 +26,7 @@ import { NCard } from 'naive-ui'
import props from './props'
import { throttle } from 'lodash-es'
import { completeSize, downloadBase64File, call, renderNode } from '@/utils'
import { setupChartTheme } from './helper'
import { setupChartTheme } from './utils'
import { APP_THEME } from '@/app-config'
import { useResizeObserver } from '@vueuse/core'
import { RMoreDropdown } from '@/components'

View File

@ -1,21 +1,19 @@
import type * as echarts from 'echarts/core' // `echarts` 核心模块
import type { PropType, VNode } from 'vue'
import type { MaybeArray } from '@/types'
import type { ECharts, SetOptionOpts } from 'echarts/core'
import type { MaybeComputedElementRef, MaybeElement } from '@vueuse/core'
import type {
LoadingOptions,
AutoResize,
ChartTheme,
} from '@/components/RChart/src/type'
import type { ECharts, SetOptionOpts } from 'echarts/core'
import type { MaybeComputedElementRef, MaybeElement } from '@vueuse/core'
import type {
EChartsExtensionInstallRegisters,
RChartPresetType,
RChartDownloadOptions,
} from './type'
} from './types'
import type { CardProps, DropdownProps, DropdownOption } from 'naive-ui'
import { loadingOptions } from './helper'
import { loadingOptions } from './utils'
const props = {
bordered: {

View File

@ -13,7 +13,7 @@ import type {
ChartThemeRawArray,
ChartThemeRawModules,
LoadingOptions,
} from '@/components/RChart/src/type'
} from '@/components/RChart/src/types'
/**
*

View File

@ -1,7 +1,7 @@
import RCollapseGrid from './src'
import collapseGridProps from './src/props'
import type * as RCollapseGridType from './src/type'
import type * as RCollapseGridType from './src/types'
import type { ExtractPublicPropTypes } from 'vue'
export type CollapseGridProps = ExtractPublicPropTypes<typeof collapseGridProps>

View File

@ -1,7 +1,7 @@
import { gridProps } from 'naive-ui'
import type { PropType } from 'vue'
import type { CollapseToggleText } from './type'
import type { CollapseToggleText } from './types'
import type { AnyFC, MaybeArray } from '@/types'
const props = {

View File

@ -1,7 +1,7 @@
import RIframe from './src'
import iframeProps from './src/props'
import type * as RIframeType from './src/type'
import type * as RIframeType from './src/types'
import type { ExtractPublicPropTypes } from 'vue'
export type IframeProps = ExtractPublicPropTypes<typeof iframeProps>

View File

@ -1,7 +1,7 @@
import RQRCode from './src'
import qrcodeProps from './src/props'
import type * as RQRCodeType from './src/type'
import type * as RQRCodeType from './src/types'
import type { ExtractPublicPropTypes } from 'vue'
export type QRCodeProps = ExtractPublicPropTypes<typeof qrcodeProps>

View File

@ -22,7 +22,7 @@ import type {
QRCodeRenderResponse,
GIFBuffer,
DownloadFilenameType,
} from './type'
} from './types'
import type { WatchStopHandle } from 'vue'
const readGIFAsArrayBuffer = (url: string): Promise<GIFBuffer> => {

View File

@ -9,7 +9,7 @@
* @remark
*/
import type { QRCodeStatus, QRCodeLevel } from './type'
import type { QRCodeStatus, QRCodeLevel } from './types'
import type { PropType, VNode } from 'vue'
import type { MaybeArray } from '@/types'
import type { Options } from 'awesome-qr'

View File

@ -1,7 +1,7 @@
import RTable from './src/Table'
import tableProps from './src/props'
import type * as RTableType from './src/type'
import type * as RTableType from './src/types'
import type { ExtractPublicPropTypes } from 'vue'
export type TableProps = ExtractPublicPropTypes<typeof tableProps>

View File

@ -24,7 +24,7 @@ import { config } from './shared'
import type { DropdownOption, DataTableInst } from 'naive-ui'
import type { ComponentSize } from '@/types'
import type { C as CType, PropsComponentPopselectKeys } from './type'
import type { C as CType, PropsComponentPopselectKeys } from './types'
export default defineComponent({
name: 'RTable',

View File

@ -26,7 +26,7 @@ import props from '../props'
import { call } from '@/utils'
import type { TreeOption, TreeDropInfo } from 'naive-ui'
import type { C } from '../type'
import type { C } from '../types'
import type { AnyFC } from '@/types'
import type { MaybeArray } from '@/types'

View File

@ -14,7 +14,7 @@ import { RIcon } from '@/components'
import { config } from '../shared'
import { useFullscreen } from 'vue-hooks-plus'
import type { TableProvider } from '../type'
import type { TableProvider } from '../types'
export default defineComponent({
name: 'TableFullscreen',

View File

@ -15,7 +15,7 @@ import { config } from '../shared'
import props from '../props'
import { printDom } from '@/utils/dom'
import type { TableProvider } from '../type'
import type { TableProvider } from '../types'
export default defineComponent({
name: 'TablePrint',

View File

@ -17,7 +17,7 @@ import { config } from '../shared'
import props from '../props'
import type { MaybeArray } from '@/types'
import type { PropsComponentPopselectKeys } from '../type'
import type { PropsComponentPopselectKeys } from '../types'
export default defineComponent({
name: 'TablePropsSelect',

View File

@ -14,7 +14,7 @@ import { dataTableProps } from 'naive-ui'
import type { PropType, VNode } from 'vue'
import type { MaybeArray } from '@/types'
import type { DropdownOption, DataTableColumn } from 'naive-ui'
import type { DownloadCsvTableOptions, PrintTableOptions } from './type'
import type { DownloadCsvTableOptions, PrintTableOptions } from './types'
import type { Recordable } from '@/types'
const props = {

View File

@ -1,7 +1,7 @@
import RTransitionComponent from './src/index.vue'
import transitionComponentProps from './src/props'
import type * as RTransitionComponentType from './src/type'
import type * as RTransitionComponentType from './src/types'
import type { ExtractPublicPropTypes } from 'vue'
export type TransitionComponentProps = ExtractPublicPropTypes<

View File

@ -27,7 +27,7 @@ import { useKeepAliveGetters } from '@/store'
import { APP_KEEP_ALIVE } from '@/app-config'
import props from './props'
import type { TransitionProps } from './type'
import type { TransitionProps } from './types'
/**
*
@ -42,3 +42,4 @@ withDefaults(defineProps<TransitionProps>(), props)
const { getKeepAliveInclude } = useKeepAliveGetters()
const { setupKeepAlive, maxKeepAliveLength, keepAliveExclude } = APP_KEEP_ALIVE
</script>
./types

View File

@ -1,4 +1,4 @@
import type { TransitionProps } from './type'
import type { TransitionProps } from './types'
const props: TransitionProps = {
transitionPropName: 'fade',

View File

@ -10,9 +10,9 @@ export * from './RTable'
export * from './RTransitionComponent'
// 导出自定义组件类型
export type * from './RChart/src/type'
export type * from './RCollapseGrid/src/type'
export type * from './RIframe/src/type'
export type * from './RQRCode/src/type'
export type * from './RTable/src/type'
export type * from './RTransitionComponent/src/type'
export type * from './RChart/src/types'
export type * from './RCollapseGrid/src/types'
export type * from './RIframe/src/types'
export type * from './RQRCode/src/types'
export type * from './RTable/src/types'
export type * from './RTransitionComponent/src/types'

View File

@ -9,11 +9,11 @@
* @remark
*/
import { combineDirective } from './helper/combine'
import { combineDirective } from './utils/combine'
import { forIn } from 'lodash-es'
import type { App } from 'vue'
import type { DirectiveModules } from '@/directives/type'
import type { DirectiveModules } from '@/directives/types'
/**
*

View File

@ -12,12 +12,22 @@
/**
*
* directive name: copy
*
* 使 value
*
* clipboard.js
*
* 使
* @example
* <template>
* <button v-copy="copyText"></button>
* </template>
*/
import ClipboardJS from 'clipboard'
import type { CopyElement } from './type'
import type { CustomDirectiveFC } from '@/directives/type'
import type { CopyElement } from './types'
import type { CustomDirectiveFC } from '@/directives/types'
const createClipboard = (el: CopyElement, value: string) => {
const clipboard = new ClipboardJS(el, {

View File

@ -12,15 +12,28 @@
/**
*
* directive name: debounce
*
* 使 func
*
* trigger wait trigger clickwait 500
*
* 使
* @example
* <template>
* <div v-debounce="{ func: () => console.log('debounce') }"></div>
* </template>
* <template>
* <div v-debounce="{ func: () => console.log('debounce'), trigger: 'click', wait: 500 }"></div>
* </template>
*/
import { debounce } from 'lodash-es'
import { useEventListener } from '@vueuse/core'
import type { DebounceBindingOptions } from './type'
import type { DebounceBindingOptions } from './types'
import type { AnyFC } from '@/types'
import type { DebouncedFunc } from 'lodash-es'
import type { CustomDirectiveFC } from '@/directives/type'
import type { CustomDirectiveFC } from '@/directives/types'
const debounceDirective: CustomDirectiveFC<
HTMLElement,

View File

@ -12,11 +12,22 @@
/**
*
* directive name: disabled
*
* 使 value
*
* ray-template__directive--disabled
* css
*
* 使
* @example
* <template>
* <button v-disabled="true"></button>
* </template>
*/
import { setClass, removeClass } from '@/utils'
import type { CustomDirectiveFC } from '@/directives/type'
import type { CustomDirectiveFC } from '@/directives/types'
const updateElementDisabledType = (el: HTMLElement, value: boolean) => {
if (el) {

View File

@ -0,0 +1,80 @@
/**
*
* directive name: ellipsis
*
* 使 width
*
* line type line 1type block
*
* 使
* @example
* <template>
* <div v-ellipsis="{ line: 2, width: 200 }"></div>
* </template>
* <template>
* <div v-ellipsis="{ type: 'block', width: 200 }"></div>
* </template>
*/
import { setStyle, completeSize } from '@/utils'
import type { CustomDirectiveFC } from '@/directives/types'
import type { EllipsisBindingValue } from './types'
/**
*
* @param el
* @param options
*
* @description
*
*
* @example
* bindEllipsis(el, { line: 2, width: 200 })
*/
const bindEllipsis = (el: HTMLElement, options: EllipsisBindingValue) => {
const { line = 1, type = 'block', width, popoverText } = options
if (width === void 0 || width === null) {
console.error(`[v-ellipsis]: Expected width, but got ${width}!`)
return
}
if (popoverText) {
el.setAttribute('title', el.textContent || '')
}
if (type === 'line') {
setStyle(el, {
display: '-webkit-box',
'-webkit-box-orient': 'vertical',
'-webkit-line-clamp': line,
overflow: 'hidden',
width: completeSize(width),
})
} else {
setStyle(el, {
overflow: 'hidden',
textOverflow: 'ellipsis',
whiteSpace: 'nowrap',
width: completeSize(width),
})
}
}
const ellipsisDirective: CustomDirectiveFC<
HTMLElement,
EllipsisBindingValue
> = () => {
return {
mounted: (el, { value }) => {
bindEllipsis(el, value)
},
updated: (el, { value }) => {
bindEllipsis(el, value)
},
}
}
export default ellipsisDirective

View File

@ -0,0 +1,26 @@
export interface EllipsisBindingValue {
/**
* @description
*
*/
line: number
/**
* @description
*
*
* line: 多行省略
* block: 单行省略
*/
type: 'line' | 'block'
/**
* @description
*
*/
width: string | number
/**
*
* @description
* title
*/
popoverText?: boolean
}

View File

@ -12,15 +12,28 @@
/**
*
* directive name: throttle
*
* 使 func
*
* trigger wait trigger clickwait 500
*
* 使
* @example
* <template>
* <div v-throttle="{ func: () => console.log('throttle') }"></div>
* </template>
* <template>
* <div v-throttle="{ func: () => console.log('throttle'), trigger: 'click', wait: 500 }"></div>
* </template>
*/
import { throttle } from 'lodash-es'
import { useEventListener } from '@vueuse/core'
import type { ThrottleBindingOptions } from './type'
import type { ThrottleBindingOptions } from './types'
import type { AnyFC } from '@/types'
import type { DebouncedFunc } from 'lodash-es'
import type { CustomDirectiveFC } from '@/directives/type'
import type { CustomDirectiveFC } from '@/directives/types'
const throttleDirective: CustomDirectiveFC<
HTMLElement,

View File

@ -1,8 +1,8 @@
import type { Directive } from 'vue'
import type { App } from 'vue'
export type { DebounceBindingOptions } from './modules/debounce/type'
export type { ThrottleBindingOptions } from './modules/throttle/type'
export type { DebounceBindingOptions } from './modules/debounce/types'
export type { ThrottleBindingOptions } from './modules/throttle/types'
export type CustomDirectiveFC<T, K> = () => Directive<T, K>

View File

@ -9,7 +9,7 @@
* @remark
*/
import type { DirectiveModules, CustomDirectiveFC } from '@/directives/type'
import type { DirectiveModules, CustomDirectiveFC } from '@/directives/types'
export const combineDirective = <
T extends Record<string, DirectiveModules>,

View File

@ -21,7 +21,7 @@ import './index.scss'
import { NResult, NButton } from 'naive-ui'
import { redirectRouterToDashboard } from '@/router/helper/routerCopilot'
import { redirectRouterToDashboard } from '@/router/utils/routerCopilot'
import { resultProps } from 'naive-ui'
const PageResult = defineComponent({

View File

@ -1,9 +1,9 @@
import { useMenuGetters, depthSearchAppMenu } from '@/store'
import { isValueType } from '@/utils'
import { createMenuExtra } from '@/store/modules/menu/helper'
import { createMenuExtra } from '@/store/modules/menu/utils'
import type { AppMenuOption } from '@/types'
import type { AppMenuExtraOptions } from '@/router/type'
import type { AppMenuExtraOptions } from '@/router/types'
export type BadgeKey = string | AppMenuOption

View File

@ -23,6 +23,9 @@ export interface UseDeviceOptions extends UseWindowSizeOptions {}
/**
*
* @param options
*
* @description
*
* 768px
*

View File

@ -19,7 +19,8 @@ export type ImageType = keyof typeof domToImageMethods
export type DomToImageResult = string | Blob | Uint8ClampedArray | undefined
export interface UseDomToImageOptions extends ReDomToImageOptions {
export interface UseDomToImageOptions<T extends TargetType = Element>
extends ReDomToImageOptions {
/**
*
*
@ -37,9 +38,7 @@ export interface UseDomToImageOptions extends ReDomToImageOptions {
*
* @default undefined
*/
beforeCreate?: <T extends TargetType = Element>(
element: T | null | undefined,
) => void
beforeCreate?: (element: T | null | undefined) => void
/**
*
* @param element current dom
@ -49,10 +48,7 @@ export interface UseDomToImageOptions extends ReDomToImageOptions {
*
* @default undefined
*/
created?: <T extends TargetType = Element>(
result: DomToImageResult,
element: T,
) => void
created?: (result: DomToImageResult, element: T) => void
/**
*
* @param error dom to image error

View File

@ -13,6 +13,7 @@ import { unrefElement, effectDispose, isValueType } from '@/utils'
import { useWindowSize } from '@vueuse/core'
import type { BasicTarget } from '@/types'
import type { CSSProperties } from 'vue'
export interface UseElementFullscreenOptions {
/**
@ -92,7 +93,7 @@ export const useElementFullscreen = (
backgroundColor,
zIndex,
} = options ?? {}
const cacheStyle: Partial<CSSStyleDeclaration> = {} // 缓存一些需要被覆盖的样式,例如: transition
const cacheStyle: Partial<CSSProperties> = {} // 缓存一些需要被覆盖的样式,例如: transition
let isSetup = false
const updateStyle = () => {

View File

@ -36,6 +36,9 @@ export const useI18n = (namespace?: string) => {
*
* HMR i18n
* i18n 使
*
* HMR
*
*/
if (!i18n) {
return {

View File

@ -1,5 +1,5 @@
import print from 'print-js'
import { unrefElement } from '@/utils'
import { unrefElement, omit } from '@/utils'
import type { BasicTarget } from '@/types'
@ -40,7 +40,7 @@ export const usePrint = (target: UsePrintTarget, options?: UsePrintOptions) => {
const _target = unrefElement(target as BasicTarget) || target
print({
...options,
...omit(options, ['printable']),
printable: _target,
})
}

View File

@ -11,28 +11,47 @@
import { router } from '@/router'
import type { Router } from 'vue-router'
/**
*
* @returns vue router instance
* @description
* vue-router setup 使
*
* @remark 使 vue router instance, setup 使
* 西 vue-router useRouter 使 router
* 使 useRouter
*
* 使, ... ,
* 使 setup , 使 useRouter useRoute ,
* 使 useRoute 使 useVueRouter().router.currentRoute
*
* @example
* const { router } = useVueRouter()
*
* // 使用 router
* router.push('/path')
* router.replace('/path')
*
* // 使用类似于 useRoute 的方法
* const { router: { currentRouter } } = useVueRouter()
*
* console.log(route.fullPath)
*/
export const useVueRouter = () => {
try {
if (router) {
return {
router,
}
} else {
throw new Error()
/**
*
* HMR
* router
*
* HMR
*
*/
if (!router) {
return {
router: {} as Router,
}
} catch (e) {
throw new Error(
`[useVueRouter]: An error occurred during registration of vue-router. ${e}`,
)
}
return {
router,
}
}

7
src/icons/enter.svg Normal file
View File

@ -0,0 +1,7 @@
<svg width="20" height="20" viewBox="0 0 20 20">
<g stroke="currentColor" fill="none" fill-rule="evenodd" stroke-linecap="round"
stroke-linejoin="round">
<path d="M18 3v4c0 2-2 4-4 4H2"></path>
<path d="M8 17l-6-6 6-6"></path>
</g>
</svg>

After

Width:  |  Height:  |  Size: 254 B

View File

@ -80,7 +80,7 @@ export default defineComponent({
'closeOther',
'closeCurrentPage',
] // 哪些下拉框允许禁用
let currentContextmenuIndex = -1 // 当前右键标签页索引位置
let currentContextmenuIndex = Infinity // 当前右键标签页索引位置
const iconConfig = {
size: 16,
}
@ -154,6 +154,7 @@ export default defineComponent({
})
const MENU_TAG_DATA = 'menu_tag_data' // 注入 tag 前缀
const globalMainLayoutLoad = getVariableToRefs('globalMainLayoutLoad')
const naiveScrollbarContainerClass = 'n-scrollbar-container' // naive scrollbar 容器 class
/**
*
@ -200,7 +201,7 @@ export default defineComponent({
scroll.childNodes,
) as HTMLElement[]
const findElement = scrollContentElement.find((el) => {
const has = hasClass(el, 'n-scrollbar-container')
const has = hasClass(el, naiveScrollbarContainerClass)
return has.value
})
@ -241,7 +242,11 @@ export default defineComponent({
const actionDropdownSelect = (key: string | number) => {
actionState.actionDropdownShow = false
actionMap[key]?.()
const fn = actionMap[key as keyof typeof actionMap]
if (fn) {
fn()
}
}
/**

View File

@ -52,13 +52,33 @@ $globalSearchWidth: 650px;
transform: scale(0.75);
font-weight: bolder;
}
& .global-search__card-content .content-item {
&.content-item--active,
&:hover {
background-color: var(--ray-theme-primary-fade-color);
}
}
}
}
}
.n-flex.global-search__card-content {
& .content-item.content-item--active,
& .content-item:hover {
background-color: var(--ray-theme-primary-fade-color);
}
& .content-item {
position: relative;
transition: var(--r-bezier);
& .content-item-icon__enter {
position: absolute;
width: 18px;
height: 18px;
right: 16px;
opacity: 0;
}
}
& .content-item.content-item--active,
& .content-item:hover {
& .content-item-icon__enter {
opacity: 1;
}
}
}

View File

@ -32,13 +32,13 @@ import {
import { RIcon } from '@/components'
import { queryElements, setClass, removeClass, pick } from '@/utils'
import { debounce } from 'lodash-es'
import { throttle } from 'lodash-es'
import { useMenuActions } from '@/store'
import { validMenuItemShow } from '@/router/helper/routerCopilot'
import { validMenuItemShow } from '@/router/utils/routerCopilot'
import { useDevice } from '@/hooks'
import { useEventListener } from '@vueuse/core'
import type { AppRouteMeta } from '@/router/type'
import type { AppRouteMeta } from '@/router/types'
import type { AppMenuOption } from '@/types'
export default defineComponent({
@ -288,6 +288,11 @@ export default defineComponent({
>
<div class="content-item-icon">{RenderPreIcon(menuOption.meta)}</div>
<div class="content-item-label">{menuOption.breadcrumbLabel}</div>
<RIcon
name="enter"
size="18"
customClassName="content-item-icon__enter"
/>
</NFlex>
)
@ -312,7 +317,7 @@ export default defineComponent({
...toRefs(state),
modelShow,
helperTipOptions,
fuzzySearchMenuOptions: debounce(fuzzySearchMenuOptions, 300),
fuzzySearchMenuOptions: throttle(fuzzySearchMenuOptions, 300),
searchItemClick,
RenderPreIcon,
isTabletOrSmaller,
@ -322,7 +327,7 @@ export default defineComponent({
},
render() {
const { isTabletOrSmaller, searchOptions, loading } = this
const { SearchItem, fuzzySearchMenuOptions, $t } = this
const { SearchItem, fuzzySearchMenuOptions } = this
return isTabletOrSmaller ? (
<div style="display: none;"></div>
@ -368,7 +373,7 @@ export default defineComponent({
size={[0, 6]}
class="global-search__card-content"
>
{searchOptions.map((curr) => (
{searchOptions.map((curr, idx) => (
<SearchItem menuOption={curr} key={curr.fullPath} />
))}
</NFlex>

View File

@ -22,7 +22,11 @@ import type { Ref } from 'vue'
* ,
*/
export const layoutHeaderCssVars = (
element: Ref<HTMLElement | undefined>[],
element: [
Ref<HTMLElement | undefined>,
Ref<HTMLElement | undefined>,
Ref<HTMLElement | undefined>,
],
) => {
const siderBar = useElementBounding(element[0])
const menuTag = useElementBounding(element[1])

View File

@ -1,133 +0,0 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-05-19
*
* @workspace ray-template
*
* @remark
*/
/**
*
* :
* - combineI18nMessages: 合并对应文件下语言包
* - getAppLocalMessages: 获取所有语言
*/
import { set } from 'lodash-es'
import { zhCN, dateZhCN } from 'naive-ui' // 导入 `naive ui` 中文包
import { getStorage } from '@/utils'
import { APP_CATCH_KEY, SYSTEM_DEFAULT_LOCAL } from '@/app-config'
import type { Recordable } from '@/types'
import type {
AppLocalesModules,
AppLocalesDropdownMixedOption,
AppCurrentAppMessages,
I18nModules,
} from '@/locales/type'
/**
*
* @param langs
* @param prefix
*
* @remark , prefix
*/
export const combineI18nMessages = (langs: I18nModules, prefix: string) => {
if (typeof prefix !== 'string' || !prefix.trim()) {
throw new TypeError('Expected prefix to be a non-empty string')
}
const langsGather: Record<string, Recordable> = {}
Object.keys(langs).forEach((key) => {
const langFileModule = langs[key].default
let fileName = key.replace(`./${prefix}/`, '').replace(/^\.\//, '')
const lastIndex = fileName.lastIndexOf('.')
fileName = fileName.substring(0, lastIndex)
const keyList = fileName.split('/')
const moduleName = keyList.shift()
const objKey = keyList.join('.')
if (moduleName) {
if (objKey) {
set(langsGather, moduleName, langsGather[moduleName] || {})
set(langsGather[moduleName], objKey, langFileModule)
} else {
set(langsGather, moduleName, langFileModule || {})
}
}
})
return langsGather
}
/** 获取所有语言 */
export const getAppLocalMessages = async (
localOptions: AppLocalesDropdownMixedOption[],
) => {
const message = {} as AppCurrentAppMessages
for (const curr of localOptions) {
const msg: AppLocalesModules = await import(`@/locales/lang/${curr.key}.ts`)
const key = curr.key
if (key) {
message[key] = msg?.default?.message ?? {}
}
}
return message
}
/**
*
* @param key
* @returns
*
* @remark . , (https://www.naiveui.com/zh-CN/dark/docs/i18n)
* @remark naive ui
*
* key LOCAL_OPTIONS
*/
export const naiveLocales = (key: string) => {
switch (key) {
case 'zh-CN':
return {
locale: zhCN,
dateLocal: dateZhCN,
}
case 'en-US':
return {
locale: null,
dateLocal: null,
}
default:
return {
locale: zhCN,
dateLocal: dateZhCN,
}
}
}
/**
*
* @returns
*
* @remark , `main.ts` , `i18n`
*/
export const getAppDefaultLanguage = () => {
const language = getStorage(APP_CATCH_KEY.localeLanguage, 'localStorage', {
defaultValue: SYSTEM_DEFAULT_LOCAL,
})
return language as keyof AppCurrentAppMessages
}

View File

@ -24,7 +24,7 @@
*/
import { createI18n } from 'vue-i18n'
import { getAppDefaultLanguage, getAppLocalMessages } from '@/locales/helper'
import { getAppDefaultLanguage, getAppLocalMessages } from '@/locales/utils'
import { SYSTEM_FALLBACK_LOCALE, LOCAL_OPTIONS } from '@/app-config'
import type { App } from 'vue'

View File

@ -1,6 +1,6 @@
import { combineI18nMessages } from '@/locales/helper'
import { combineI18nMessages } from '@/locales/utils'
import type { I18nModules } from '@/locales/type'
import type { I18nModules } from '@/locales/types'
const modules: I18nModules = import.meta.glob('./en-US/**/*.json', {
eager: true,

View File

@ -1,6 +1,6 @@
import { combineI18nMessages } from '@/locales/helper'
import { combineI18nMessages } from '@/locales/utils'
import type { I18nModules } from '@/locales/type'
import type { I18nModules } from '@/locales/types'
const modules: I18nModules = import.meta.glob('./zh-CN/**/*.json', {
eager: true,

View File

@ -0,0 +1,47 @@
import { set } from 'lodash-es'
import type { Recordable } from '@/types'
import type { I18nModules } from '@/locales/types'
/**
*
* @param langs
* @param prefix
*
* @description
* lang
*
* @example
* const messages = combineI18nMessages(modules, 'zh-CN')
*/
export const combineI18nMessages = (langs: I18nModules, prefix: string) => {
if (typeof prefix !== 'string' || !prefix.trim()) {
throw new TypeError('Expected prefix to be a non-empty string')
}
const langsGather: Record<string, Recordable> = {}
Object.keys(langs).forEach((key) => {
const langFileModule = langs[key].default
let fileName = key.replace(`./${prefix}/`, '').replace(/^\.\//, '')
const lastIndex = fileName.lastIndexOf('.')
fileName = fileName.substring(0, lastIndex)
const keyList = fileName.split('/')
const moduleName = keyList.shift()
const objKey = keyList.join('.')
if (moduleName) {
if (objKey) {
set(langsGather, moduleName, langsGather[moduleName] || {})
set(langsGather[moduleName], objKey, langFileModule)
} else {
set(langsGather, moduleName, langFileModule || {})
}
}
})
return langsGather
}

View File

@ -0,0 +1,20 @@
import { getStorage } from '@/utils'
import { APP_CATCH_KEY, SYSTEM_DEFAULT_LOCAL } from '@/app-config'
import type { AppCurrentAppMessages } from '@/locales/types'
/**
*
* @description
*
*
* @example
* const language = getAppDefaultLanguage() // zh-CN | en-US | ...
*/
export const getAppDefaultLanguage = () => {
const language = getStorage(APP_CATCH_KEY.localeLanguage, 'localStorage', {
defaultValue: SYSTEM_DEFAULT_LOCAL,
})
return language as keyof AppCurrentAppMessages
}

View File

@ -0,0 +1,32 @@
import type {
AppLocalesModules,
AppLocalesDropdownMixedOption,
AppCurrentAppMessages,
} from '@/locales/types'
/**
*
* @param localOptions
*
* @description
*
*
* @example
* const messages = await getAppLocalMessages(localOptions)
*/
export const getAppLocalMessages = async (
localOptions: AppLocalesDropdownMixedOption[],
) => {
const message = {} as AppCurrentAppMessages
for (const curr of localOptions) {
const msg: AppLocalesModules = await import(`@/locales/lang/${curr.key}.ts`)
const key = curr.key
if (key) {
message[key] = msg?.default?.message ?? {}
}
}
return message
}

View File

@ -0,0 +1,35 @@
import { zhCN, dateZhCN } from 'naive-ui' // 导入 `naive ui` 中文包
/**
*
* @param key
*
* @description
* key naive ui
*
* key LOCAL_OPTIONS key
*
* @example
* const { locale, dateLocal } = naiveLocales('zh-CN')
*/
export const getNaiveLocales = (key: string) => {
switch (key) {
case 'zh-CN':
return {
locale: zhCN,
dateLocal: dateZhCN,
}
case 'en-US':
return {
locale: null,
dateLocal: null,
}
default:
return {
locale: zhCN,
dateLocal: dateZhCN,
}
}
}

View File

@ -0,0 +1,4 @@
export * from './combineI18nMessages'
export * from './getAppLocalMessages'
export * from './getNaiveLocales'
export * from './getAppDefaultLanguage'

View File

@ -34,9 +34,9 @@ const setupRayTemplateCore = async () => {
*
*/
const setupPlugins = async (inst: AppType<Element>) => {
setupStore(inst)
await setupI18n(inst)
await setupStore(inst)
await setupRouter(inst)
setupRouter(inst)
setupDayjs()
setupDirectives(inst)
}

View File

@ -8,7 +8,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const demo: AppRouteRecordRaw = {
/** 路由路径,如果为根菜单且无有菜单的时候可以配置为空字符串 */

View File

@ -20,9 +20,9 @@
* order ,
*/
import { combineRawRouteModules } from '@/router/helper/setupHelper'
import { orderRoutes } from '@/router/helper/setupHelper'
import { expandRoutes } from '@/router/helper/expandRoutes'
import { combineRawRouteModules } from '@/router/utils/setupHelper'
import { orderRoutes } from '@/router/utils/setupHelper'
import { expandRoutes } from '@/router/utils/expandRoutes'
/** 获取所有被合并与排序的路由 */
export const getAppRawRoutes = () => orderRoutes(combineRawRouteModules())

View File

@ -1,35 +1,31 @@
import { createRouter, createWebHashHistory } from 'vue-router'
import { scrollViewToTop } from '@/router/helper/setupHelper'
import { vueRouterRegister } from '@/router/helper/routerCopilot'
import { scrollViewToTop } from '@/router/utils/setupHelper'
import { vueRouterRegister } from '@/router/utils/routerCopilot'
import { useVueRouter } from '@/hooks'
import constantRoutes from './routes'
import type { App } from 'vue'
import type { RouteRecordRaw, Router } from 'vue-router'
import type { RouteRecordRaw } from 'vue-router'
export let router: Router
export const router = createRouter({
history: createWebHashHistory(),
routes: constantRoutes as unknown as RouteRecordRaw[],
scrollBehavior: (to) => {
scrollViewToTop(to)
},
})
/**
*
* vue router
* scrollBehavior
* @param app vue instance
*
* @description
* vue-router
*/
const createVueRouter = async () => {
return createRouter({
history: createWebHashHistory(),
routes: (await constantRoutes()) as unknown as RouteRecordRaw[],
scrollBehavior: (to) => {
scrollViewToTop(to)
},
})
}
// setup router
export const setupRouter = async (app: App<Element>) => {
router = await createVueRouter()
export const setupRouter = (app: App<Element>) => {
app.use(router)
// 等待 router 挂载后,初始化 useRouter 方法
useVueRouter()
vueRouterRegister(router)

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const dashboard: AppRouteRecordRaw = {
path: '/dashboard',

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const axios: AppRouteRecordRaw = {
path: '/axios',

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const cacheDemo: AppRouteRecordRaw = {
path: '/cache-demo',

View File

@ -1,6 +1,6 @@
import { t } from '@/hooks'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const contextMenu: AppRouteRecordRaw = {
path: '/context-menu',

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const directive: AppRouteRecordRaw = {
path: '/directive',
@ -11,6 +11,9 @@ const directive: AppRouteRecordRaw = {
i18nKey: t('menu.Directive'),
icon: 'other',
order: 2,
extra: {
label: 'ellipsis',
},
},
}

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const doc: AppRouteRecordRaw = {
path: '/doc',

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const echart: AppRouteRecordRaw = {
path: '/echart',

View File

@ -1,6 +1,6 @@
import { t } from '@/hooks'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const iframe: AppRouteRecordRaw = {
path: '/iframe',

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const mockDemo: AppRouteRecordRaw = {
path: '/mock-demo',

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const mockDemo: AppRouteRecordRaw = {
path: '/modal-demo',

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const multiMenu: AppRouteRecordRaw = {
path: '/multi',

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const precision: AppRouteRecordRaw = {
path: '/precision',

View File

@ -1,7 +1,7 @@
import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type'
import type { AppRouteRecordRaw } from '@/router/types'
const qrcode: AppRouteRecordRaw = {
path: '/qrcode',

Some files were not shown because too many files have changed in this diff Show More