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", "domtoimage",
"EDITMSG", "EDITMSG",
"iife", "iife",
"linebreak",
"macarons", "macarons",
"menutag", "menutag",
"ndata", "ndata",
@ -32,6 +33,7 @@
"Popselect", "Popselect",
"precommit", "precommit",
"siderbar", "siderbar",
"stylelint",
"WUJIE", "WUJIE",
"zlevel" "zlevel"
] ]

View File

@ -1,5 +1,48 @@
# CHANGE LOG # 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
稳定了 `4.6.4` 版本。 稳定了 `4.6.4` 版本。
@ -208,7 +251,7 @@ remove('your key', 'all')
import { t } from '@/hooks' import { t } from '@/hooks'
import { LAYOUT } from '@/router/constant' import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/type' import type { AppRouteRecordRaw } from '@/router/types'
const cacheDemo: AppRouteRecordRaw = { const cacheDemo: AppRouteRecordRaw = {
// ...your route config, // ...your route config,

View File

@ -1,7 +1,7 @@
{ {
"name": "ray-template", "name": "ray-template",
"private": false, "private": false,
"version": "4.6.4", "version": "4.7.0",
"type": "module", "type": "module",
"engines": { "engines": {
"node": "^18.0.0 || >=20.0.0", "node": "^18.0.0 || >=20.0.0",
@ -32,7 +32,7 @@
] ]
}, },
"dependencies": { "dependencies": {
"@vueuse/core": "^10.7.1", "@vueuse/core": "^10.9.0",
"awesome-qr": "2.1.5-rc.0", "awesome-qr": "2.1.5-rc.0",
"axios": "^1.6.7", "axios": "^1.6.7",
"clipboard": "^2.0.11", "clipboard": "^2.0.11",
@ -48,7 +48,7 @@
"pinia": "^2.1.7", "pinia": "^2.1.7",
"pinia-plugin-persistedstate": "^3.2.0", "pinia-plugin-persistedstate": "^3.2.0",
"print-js": "^1.6.0", "print-js": "^1.6.0",
"vue": "^3.4.20", "vue": "^3.4.21",
"vue-demi": "0.14.6", "vue-demi": "0.14.6",
"vue-hooks-plus": "1.8.8", "vue-hooks-plus": "1.8.8",
"vue-i18n": "^9.9.0", "vue-i18n": "^9.9.0",
@ -87,13 +87,13 @@
"postcss": "^8.4.31", "postcss": "^8.4.31",
"postcss-px-to-viewport-8-plugin": "1.2.3", "postcss-px-to-viewport-8-plugin": "1.2.3",
"prettier": "^3.2.5", "prettier": "^3.2.5",
"rollup-plugin-visualizer": "^5.12.0",
"sass": "1.71.1", "sass": "1.71.1",
"svg-sprite-loader": "^6.0.11", "svg-sprite-loader": "^6.0.11",
"typescript": "^5.2.2", "typescript": "^5.2.2",
"unplugin-auto-import": "^0.17.5", "unplugin-auto-import": "^0.17.5",
"unplugin-vue-components": "^0.26.0", "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-cdn2": "0.15.4",
"vite-plugin-compression": "^0.5.1", "vite-plugin-compression": "^0.5.1",
"vite-plugin-ejs": "^1.7.0", "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` 是否配置正确 - validAppRootPath: 检查模板 `appRootPath` 是否配置正确
- validLocal: 检查模板 `localConfig` 是否配置正确 - validLocal: 检查模板 `localConfig` 是否配置正确
## 拓展
当你需要在做一些定制化操作的时候,可以尝试在这个包里做一些事情。
租后在 `main.ts` 中导入并且调用即可。
> 出于一些考虑,并没有做自动化导入调用,所以需要自己手动来。(好吧,其实就是我懒--

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

@ -6,6 +6,7 @@ import type {
AxiosDefaults, AxiosDefaults,
Axios, Axios,
AxiosResponse, AxiosResponse,
AxiosError,
} from 'axios' } from 'axios'
import type { AnyFC } from '@/types' import type { AnyFC } from '@/types'
@ -28,8 +29,13 @@ export interface CancelConfig {
export interface AppRawRequestConfig<T = any> extends AxiosRequestConfig<T> { export interface AppRawRequestConfig<T = any> extends AxiosRequestConfig<T> {
cancelConfig?: CancelConfig 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 { export interface AxiosInstanceExpand extends Axios {
<T = any, D = any>(config: AppRawRequestConfig<D>): Promise<T> <T = any, D = any>(config: AppRawRequestConfig<D>): Promise<T>
<T = any, D = any>(url: string, 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[] implementResponseInterceptorErrorArray: AnyFC[]
} }
export type BeforeFetchFunction< export type FetchFunction = <T = any, K = any>(
T = RequestInterceptorConfig | ResponseInterceptorConfig, ins: RequestInterceptorConfig<T> & ResponseInterceptorConfig<T, K>,
> = <K extends T>(ins: K, mode: string) => void mode: string,
) => void
export type FetchType = 'ok' | 'error' export type FetchType = 'ok' | 'error'
export type FetchErrorFunction<T = any> = <K extends T>( export type FetchErrorFunction<T = any, D = any> = (
error: K, error: AxiosError<T, D>,
mode: string, mode: string,
) => void ) => void
@ -120,7 +127,7 @@ export interface AxiosFetchInstance {
responseInstance: ResponseInterceptorConfig | null responseInstance: ResponseInterceptorConfig | null
} }
export interface AxiosFetchError<T = unknown> { export interface AxiosFetchError<T = any> {
requestError: T | null requestError: T | null
responseError: 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 { export default class RequestCanceler {
private pendingRequest: Map<string, AbortController> 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 return config.cancelConfig?.cancel ?? true
} }
@ -37,7 +37,7 @@ export default class RequestCanceler {
* *
* @remark config request key * @remark config request key
*/ */
private generateRequestKey(config: AppRawRequestConfig) { private generateRequestKey(config: AppRawRequestConfig | CancelerParams) {
const { method, url } = config const { method, url } = config
return [ return [
@ -54,8 +54,10 @@ export default class RequestCanceler {
* *
* @remark signal , * @remark signal ,
*/ */
addPendingRequest(config: AppRawRequestConfig) { addPendingRequest(config: AppRawRequestConfig | CancelerParams) {
if (this.isAppending(config)) { if (this.isAppending(config)) {
config.__CANCELER_TAG_RAY_TEMPLATE__ = '__CANCELER_TAG_RAY_TEMPLATE__'
const requestKey = this.generateRequestKey(config) const requestKey = this.generateRequestKey(config)
if (!this.pendingRequest.has(requestKey)) { if (!this.pendingRequest.has(requestKey)) {
@ -77,7 +79,7 @@ export default class RequestCanceler {
* *
* @remark , map generateRequestKey value * @remark , map generateRequestKey value
*/ */
removePendingRequest(config: AppRawRequestConfig) { removePendingRequest(config: AppRawRequestConfig | CancelerParams) {
const requestKey = this.generateRequestKey(config) const requestKey = this.generateRequestKey(config)
if (this.pendingRequest.has(requestKey)) { if (this.pendingRequest.has(requestKey)) {

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

@ -1,7 +1,7 @@
import RTable from './src/Table' import RTable from './src/Table'
import tableProps from './src/props' import tableProps from './src/props'
import type * as RTableType from './src/type' import type * as RTableType from './src/types'
import type { ExtractPublicPropTypes } from 'vue' import type { ExtractPublicPropTypes } from 'vue'
export type TableProps = ExtractPublicPropTypes<typeof tableProps> 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 { DropdownOption, DataTableInst } from 'naive-ui'
import type { ComponentSize } from '@/types' import type { ComponentSize } from '@/types'
import type { C as CType, PropsComponentPopselectKeys } from './type' import type { C as CType, PropsComponentPopselectKeys } from './types'
export default defineComponent({ export default defineComponent({
name: 'RTable', name: 'RTable',

View File

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

View File

@ -14,7 +14,7 @@ import { RIcon } from '@/components'
import { config } from '../shared' 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 '../types'
export default defineComponent({ export default defineComponent({
name: 'TableFullscreen', name: 'TableFullscreen',

View File

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

View File

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

View File

@ -14,7 +14,7 @@ import { dataTableProps } from 'naive-ui'
import type { PropType, VNode } 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 { DownloadCsvTableOptions, PrintTableOptions } from './type' import type { DownloadCsvTableOptions, PrintTableOptions } from './types'
import type { Recordable } from '@/types' import type { Recordable } from '@/types'
const props = { const props = {

View File

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

View File

@ -27,7 +27,7 @@ import { useKeepAliveGetters } from '@/store'
import { APP_KEEP_ALIVE } from '@/app-config' import { APP_KEEP_ALIVE } from '@/app-config'
import props from './props' 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 { getKeepAliveInclude } = useKeepAliveGetters()
const { setupKeepAlive, maxKeepAliveLength, keepAliveExclude } = APP_KEEP_ALIVE const { setupKeepAlive, maxKeepAliveLength, keepAliveExclude } = APP_KEEP_ALIVE
</script> </script>
./types

View File

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

View File

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

View File

@ -9,11 +9,11 @@
* @remark * @remark
*/ */
import { combineDirective } from './helper/combine' import { combineDirective } from './utils/combine'
import { forIn } from 'lodash-es' import { forIn } from 'lodash-es'
import type { App } from 'vue' 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 * directive name: copy
*
* 使 value
*
* clipboard.js
*
* 使
* @example
* <template>
* <button v-copy="copyText"></button>
* </template>
*/ */
import ClipboardJS from 'clipboard' import ClipboardJS from 'clipboard'
import type { CopyElement } from './type' import type { CopyElement } from './types'
import type { CustomDirectiveFC } from '@/directives/type' import type { CustomDirectiveFC } from '@/directives/types'
const createClipboard = (el: CopyElement, value: string) => { const createClipboard = (el: CopyElement, value: string) => {
const clipboard = new ClipboardJS(el, { const clipboard = new ClipboardJS(el, {

View File

@ -12,15 +12,28 @@
/** /**
* *
* directive name: debounce * 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 { debounce } from 'lodash-es'
import { useEventListener } from '@vueuse/core' import { useEventListener } from '@vueuse/core'
import type { DebounceBindingOptions } from './type' import type { DebounceBindingOptions } from './types'
import type { AnyFC } from '@/types' import type { AnyFC } from '@/types'
import type { DebouncedFunc } from 'lodash-es' import type { DebouncedFunc } from 'lodash-es'
import type { CustomDirectiveFC } from '@/directives/type' import type { CustomDirectiveFC } from '@/directives/types'
const debounceDirective: CustomDirectiveFC< const debounceDirective: CustomDirectiveFC<
HTMLElement, HTMLElement,

View File

@ -12,11 +12,22 @@
/** /**
* *
* directive name: disabled * directive name: disabled
*
* 使 value
*
* ray-template__directive--disabled
* css
*
* 使
* @example
* <template>
* <button v-disabled="true"></button>
* </template>
*/ */
import { setClass, removeClass } from '@/utils' import { setClass, removeClass } from '@/utils'
import type { CustomDirectiveFC } from '@/directives/type' import type { CustomDirectiveFC } from '@/directives/types'
const updateElementDisabledType = (el: HTMLElement, value: boolean) => { const updateElementDisabledType = (el: HTMLElement, value: boolean) => {
if (el) { 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 * 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 { throttle } from 'lodash-es'
import { useEventListener } from '@vueuse/core' import { useEventListener } from '@vueuse/core'
import type { ThrottleBindingOptions } from './type' import type { ThrottleBindingOptions } from './types'
import type { AnyFC } from '@/types' import type { AnyFC } from '@/types'
import type { DebouncedFunc } from 'lodash-es' import type { DebouncedFunc } from 'lodash-es'
import type { CustomDirectiveFC } from '@/directives/type' import type { CustomDirectiveFC } from '@/directives/types'
const throttleDirective: CustomDirectiveFC< const throttleDirective: CustomDirectiveFC<
HTMLElement, HTMLElement,

View File

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

View File

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

View File

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

View File

@ -1,9 +1,9 @@
import { useMenuGetters, depthSearchAppMenu } from '@/store' import { useMenuGetters, depthSearchAppMenu } from '@/store'
import { isValueType } from '@/utils' 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 { AppMenuOption } from '@/types'
import type { AppMenuExtraOptions } from '@/router/type' import type { AppMenuExtraOptions } from '@/router/types'
export type BadgeKey = string | AppMenuOption export type BadgeKey = string | AppMenuOption

View File

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

View File

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

View File

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

View File

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

View File

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

View File

@ -11,28 +11,47 @@
import { router } from '@/router' 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
* *
* 使, ... , * 使 useRoute 使 useVueRouter().router.currentRoute
* 使 setup , 使 useRouter useRoute , *
* @example
* const { router } = useVueRouter()
*
* // 使用 router
* router.push('/path')
* router.replace('/path')
*
* // 使用类似于 useRoute 的方法
* const { router: { currentRouter } } = useVueRouter()
*
* console.log(route.fullPath)
*/ */
export const useVueRouter = () => { export const useVueRouter = () => {
try { /**
if (router) { *
return { * HMR
router, * router
} *
} else { * HMR
throw new Error() *
*/
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', 'closeOther',
'closeCurrentPage', 'closeCurrentPage',
] // 哪些下拉框允许禁用 ] // 哪些下拉框允许禁用
let currentContextmenuIndex = -1 // 当前右键标签页索引位置 let currentContextmenuIndex = Infinity // 当前右键标签页索引位置
const iconConfig = { const iconConfig = {
size: 16, size: 16,
} }
@ -154,6 +154,7 @@ export default defineComponent({
}) })
const MENU_TAG_DATA = 'menu_tag_data' // 注入 tag 前缀 const MENU_TAG_DATA = 'menu_tag_data' // 注入 tag 前缀
const globalMainLayoutLoad = getVariableToRefs('globalMainLayoutLoad') const globalMainLayoutLoad = getVariableToRefs('globalMainLayoutLoad')
const naiveScrollbarContainerClass = 'n-scrollbar-container' // naive scrollbar 容器 class
/** /**
* *
@ -200,7 +201,7 @@ export default defineComponent({
scroll.childNodes, scroll.childNodes,
) as HTMLElement[] ) as HTMLElement[]
const findElement = scrollContentElement.find((el) => { const findElement = scrollContentElement.find((el) => {
const has = hasClass(el, 'n-scrollbar-container') const has = hasClass(el, naiveScrollbarContainerClass)
return has.value return has.value
}) })
@ -241,7 +242,11 @@ export default defineComponent({
const actionDropdownSelect = (key: string | number) => { const actionDropdownSelect = (key: string | number) => {
actionState.actionDropdownShow = false 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); transform: scale(0.75);
font-weight: bolder; 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 { RIcon } from '@/components'
import { queryElements, setClass, removeClass, pick } from '@/utils' import { queryElements, setClass, removeClass, pick } from '@/utils'
import { debounce } from 'lodash-es' import { throttle } from 'lodash-es'
import { useMenuActions } from '@/store' import { useMenuActions } from '@/store'
import { validMenuItemShow } from '@/router/helper/routerCopilot' import { validMenuItemShow } from '@/router/utils/routerCopilot'
import { useDevice } from '@/hooks' import { useDevice } from '@/hooks'
import { useEventListener } from '@vueuse/core' import { useEventListener } from '@vueuse/core'
import type { AppRouteMeta } from '@/router/type' import type { AppRouteMeta } from '@/router/types'
import type { AppMenuOption } from '@/types' import type { AppMenuOption } from '@/types'
export default defineComponent({ export default defineComponent({
@ -288,6 +288,11 @@ export default defineComponent({
> >
<div class="content-item-icon">{RenderPreIcon(menuOption.meta)}</div> <div class="content-item-icon">{RenderPreIcon(menuOption.meta)}</div>
<div class="content-item-label">{menuOption.breadcrumbLabel}</div> <div class="content-item-label">{menuOption.breadcrumbLabel}</div>
<RIcon
name="enter"
size="18"
customClassName="content-item-icon__enter"
/>
</NFlex> </NFlex>
) )
@ -312,7 +317,7 @@ export default defineComponent({
...toRefs(state), ...toRefs(state),
modelShow, modelShow,
helperTipOptions, helperTipOptions,
fuzzySearchMenuOptions: debounce(fuzzySearchMenuOptions, 300), fuzzySearchMenuOptions: throttle(fuzzySearchMenuOptions, 300),
searchItemClick, searchItemClick,
RenderPreIcon, RenderPreIcon,
isTabletOrSmaller, isTabletOrSmaller,
@ -322,7 +327,7 @@ export default defineComponent({
}, },
render() { render() {
const { isTabletOrSmaller, searchOptions, loading } = this const { isTabletOrSmaller, searchOptions, loading } = this
const { SearchItem, fuzzySearchMenuOptions, $t } = this const { SearchItem, fuzzySearchMenuOptions } = this
return isTabletOrSmaller ? ( return isTabletOrSmaller ? (
<div style="display: none;"></div> <div style="display: none;"></div>
@ -368,7 +373,7 @@ export default defineComponent({
size={[0, 6]} size={[0, 6]}
class="global-search__card-content" class="global-search__card-content"
> >
{searchOptions.map((curr) => ( {searchOptions.map((curr, idx) => (
<SearchItem menuOption={curr} key={curr.fullPath} /> <SearchItem menuOption={curr} key={curr.fullPath} />
))} ))}
</NFlex> </NFlex>

View File

@ -22,7 +22,11 @@ import type { Ref } from 'vue'
* , * ,
*/ */
export const layoutHeaderCssVars = ( export const layoutHeaderCssVars = (
element: Ref<HTMLElement | undefined>[], element: [
Ref<HTMLElement | undefined>,
Ref<HTMLElement | undefined>,
Ref<HTMLElement | undefined>,
],
) => { ) => {
const siderBar = useElementBounding(element[0]) const siderBar = useElementBounding(element[0])
const menuTag = useElementBounding(element[1]) 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 { 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 { SYSTEM_FALLBACK_LOCALE, LOCAL_OPTIONS } from '@/app-config'
import type { App } from 'vue' 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', { const modules: I18nModules = import.meta.glob('./en-US/**/*.json', {
eager: true, 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', { const modules: I18nModules = import.meta.glob('./zh-CN/**/*.json', {
eager: true, 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>) => { const setupPlugins = async (inst: AppType<Element>) => {
setupStore(inst)
await setupI18n(inst) await setupI18n(inst)
await setupStore(inst) setupRouter(inst)
await setupRouter(inst)
setupDayjs() setupDayjs()
setupDirectives(inst) setupDirectives(inst)
} }

View File

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

View File

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

View File

@ -1,35 +1,31 @@
import { createRouter, createWebHashHistory } from 'vue-router' import { createRouter, createWebHashHistory } from 'vue-router'
import { scrollViewToTop } from '@/router/helper/setupHelper' import { scrollViewToTop } from '@/router/utils/setupHelper'
import { vueRouterRegister } from '@/router/helper/routerCopilot' import { vueRouterRegister } from '@/router/utils/routerCopilot'
import { useVueRouter } from '@/hooks' import { useVueRouter } from '@/hooks'
import constantRoutes from './routes' import constantRoutes from './routes'
import type { App } from 'vue' 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 * @param app vue instance
* scrollBehavior *
* @description
* vue-router
*/ */
const createVueRouter = async () => { export const setupRouter = (app: App<Element>) => {
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()
app.use(router) app.use(router)
// 等待 router 挂载后,初始化 useRouter 方法 // 等待 router 挂载后,初始化 useRouter 方法
useVueRouter() useVueRouter()
vueRouterRegister(router) vueRouterRegister(router)

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

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