version: v5.0.4

This commit is contained in:
XiaoDaiGua-Ray 2024-11-09 14:55:41 +08:00
parent 8f3969268a
commit ff1a67c843
56 changed files with 7180 additions and 5178 deletions

View File

@ -297,5 +297,20 @@ module.exports = {
next: ['while', 'do', 'switch'],
},
],
'@typescript-eslint/no-unused-expressions': [
'error',
{
allowShortCircuit: true,
allowTernary: true,
allowTaggedTemplates: true,
},
],
'@typescript-eslint/no-empty-object-type': [
'error',
{
allowInterfaces: 'with-single-extends',
allowObjectTypes: 'always',
},
],
},
}

View File

@ -32,6 +32,7 @@
"internalkey",
"jsbarcode",
"linebreak",
"logicflow",
"macarons",
"menutag",
"ndata",
@ -40,6 +41,7 @@
"Popselect",
"precommit",
"siderbar",
"snapline",
"stylelint",
"WUJIE",
"zlevel"

View File

@ -1,5 +1,33 @@
# CHANGE LOG
## 5.0.4
`ts` 版本与 `eslint` 解析插件版本更新至最新版,并且同步解决了以前历史遗留的一些问题。
并且,在该版本做了一些全局注入方式调整,请谨慎更新。
## Feats
- 移除 `vite-plugin-compression` 插件,使用 `rollup-plugin-gzip` 代替
- 更新 `@vitejs/plugin-vue-jsx` 版本至 `4.0.1`
- 优化注释
- 默认设置 `ContentWrapper``content-wrapper` 内容展示区域的宽高为 `100%`,继承父容器的宽高;该样式会自动的计算,也就是说会自动的适配不同尺寸的屏幕输出与判断是否显示 `FeatureWrapper`, `FooterWrapper`
- 更新 `@typescript-eslint/eslint-plugin`, `@typescript-eslint/parser` 版本至 `8.13.0`
- 更新 `typescript` 版本至 `5.6.3`
- 更新 `vue-tsc` 版本至 `2.1.10`
- 更新 `pnpm` 包管理器版本至 `9.12.3`
- 新增 `RFlow` 基础流程图组件(后期有时间会逐步加强该组件)
- 移除 `useElementFullscreen` 方法 `currentWindowSize` 返回值
- 新增 `--html-height`, `--html-width` 的全局 `css var` 属性,实时获取浏览器的尺寸
- 样式注入现在由注入至 `body` 改为注入至 `html`,避免 `teleport` 传送至 `body` 外的元素不能使用全局样式的问题
- `useBadge.show` 方法新增 `extraOption` 配置项,允许在显示 `badge` 到时候,传入额外的 `options` 配置项
## Fixes
- 修复菜单折叠后,`SiderBarLogo` 标题样式丢失问题
- 修复 `useModal` 因为 `Omit` 原因导致类型丢失问题,现在直接使用 `ModalProps` 作为类型
- 修复 `useModal` 创建全屏 `card` 的时候,内容区域边距样式被覆盖的问题,现在会尊重原有的 `card` 样式
## 5.0.3
个性化配置能力再次提升。

View File

@ -17,8 +17,8 @@ describe('useDayjs', () => {
}
const localSpy = vi.spyOn(m, 'locale')
m.locale('en')
m.locale('zh-cn')
m.locale('en-US')
m.locale('zh-CN')
expect(localSpy).toHaveBeenCalledTimes(2)
})

View File

@ -1,11 +1,11 @@
{
"name": "ray-template",
"private": false,
"version": "5.0.3",
"version": "5.0.4",
"type": "module",
"engines": {
"node": "^18.0.0 || >=20.0.0",
"pnpm": ">=8.0.0"
"pnpm": ">=9.0.0"
},
"scripts": {
"dev": "vite",
@ -33,6 +33,8 @@
]
},
"dependencies": {
"@logicflow/core": "2.0.6",
"@logicflow/extension": "2.0.10",
"@vueuse/core": "^11.1.0",
"axios": "^1.7.5",
"clipboard": "^2.0.11",
@ -61,14 +63,13 @@
"@interactjs/types": "1.10.21",
"@intlify/unplugin-vue-i18n": "^4.0.0",
"@types/crypto-js": "^4.2.2",
"@types/dom-to-image": "2.6.7",
"@types/jsbarcode": "3.11.4",
"@types/lodash-es": "^4.17.12",
"@types/mockjs": "1.0.7",
"@typescript-eslint/eslint-plugin": "^6.21.0",
"@typescript-eslint/parser": "^6.21.0",
"@typescript-eslint/eslint-plugin": "^8.13.0",
"@typescript-eslint/parser": "^8.13.0",
"@vitejs/plugin-vue": "^5.1.0",
"@vitejs/plugin-vue-jsx": "^4.0.0",
"@vitejs/plugin-vue-jsx": "^4.0.1",
"@vitest/ui": "1.4.0",
"@vue/eslint-config-prettier": "^9.0.0",
"@vue/eslint-config-typescript": "^12.0.0",
@ -87,15 +88,15 @@
"postcss": "^8.4.38",
"postcss-px-to-viewport-8-with-include": "1.2.2",
"prettier": "^3.2.5",
"rollup-plugin-gzip": "4.0.1",
"sass": "1.71.1",
"svg-sprite-loader": "^6.0.11",
"typescript": "^5.2.2",
"typescript": "^5.6.3",
"unplugin-auto-import": "^0.18.2",
"unplugin-vue-components": "^0.27.4",
"vite": "^5.4.3",
"vite-bundle-analyzer": "0.9.4",
"vite-plugin-cdn2": "1.1.0",
"vite-plugin-compression": "^0.5.1",
"vite-plugin-ejs": "^1.7.0",
"vite-plugin-eslint": "1.8.1",
"vite-plugin-inspect": "^0.8.3",
@ -104,7 +105,7 @@
"vite-svg-loader": "^4.0.0",
"vite-tsconfig-paths": "4.3.2",
"vitest": "1.5.2",
"vue-tsc": "^2.0.13"
"vue-tsc": "^2.1.10"
},
"description": "<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->",
"main": "index.ts",

11403
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@ -7,7 +7,8 @@ import {
getStorage,
} from '@/utils'
import { useSettingGetters } from '@/store'
import { APP_CATCH_KEY } from '@/app-config'
import { APP_CATCH_KEY, THEME_CLASS_NAMES } from '@/app-config'
import { useWindowSize } from '@vueuse/core'
import type { SettingState } from '@/store/modules/setting/types'
@ -15,39 +16,43 @@ export default defineComponent({
name: 'AppStyleProvider',
setup(_, { expose }) {
const { getAppTheme } = useSettingGetters()
const { height, width } = useWindowSize()
/** 同步主题色变量至 body, 如果未获取到缓存值则已默认值填充 */
// 同步主题色变量至 html如果未获取到缓存值则已默认值填充
const syncPrimaryColorToBody = () => {
const {
appPrimaryColor: { primaryColor, primaryFadeColor },
} = __APP_CFG__ // 默认主题色
const body = document.body
const html = document.documentElement
// 获取缓存 naive ui 配置项
const primaryColorOverride = getStorage<SettingState>(
APP_CATCH_KEY.appPiniaSettingStore,
'localStorage',
) // 获取缓存 naive ui 配置项
)
if (primaryColorOverride) {
// 获取主色调
const p = get(
primaryColorOverride,
'primaryColorOverride.common.primaryColor',
primaryColor,
) // 获取主色调
const fp = colorToRgba(p, 0.38) // 将主色调任意颜色转换为 rgba 格式
)
// 将主色调任意颜色转换为 rgba 格式
const fp = colorToRgba(p, 0.38)
/** 设置全局主题色 css 变量 */
body.style.setProperty('--ray-theme-primary-color', p) // 主色调
body.style.setProperty(
// 设置全局主题色 css 变量
html.style.setProperty('--ray-theme-primary-color', p) // 主色调
html.style.setProperty(
'--ray-theme-primary-fade-color',
fp || primaryFadeColor,
) // 降低透明度后的主色调
}
}
/** 隐藏加载动画 */
// 隐藏加载动画
const hiddenLoadingAnimation = () => {
/** pre-loading-animation 是默认 id */
// pre-loading-animation 是默认 id
const el = document.getElementById('pre-loading-animation')
if (el) {
@ -57,38 +62,30 @@ export default defineComponent({
}
}
/** 切换主题时, 同步更新 body class 以便于进行自定义 css 配置 */
// 切换主题时,同步更新 html class 以便于进行自定义 css 配置
const updateGlobalThemeClass = (bool: boolean) => {
/**
*
* body class
*
* getAppTheme
*/
const body = document.body
const darkClassName = 'ray-template--dark' // 暗色类名
const lightClassName = 'ray-template--light' // 明亮色类名
const html = document.documentElement
const { darkClassName, lightClassName } = THEME_CLASS_NAMES
bool
? removeClass(body, lightClassName)
: removeClass(body, darkClassName)
? removeClass(html, lightClassName)
: removeClass(html, darkClassName)
setClass(body, bool ? darkClassName : lightClassName)
setClass(html, bool ? darkClassName : lightClassName)
}
syncPrimaryColorToBody()
hiddenLoadingAnimation()
// 当切换主题时,更新 body 当前的注入 class
watch(
() => getAppTheme.value,
(ndata) => {
updateGlobalThemeClass(ndata)
},
{
immediate: true,
},
)
watchEffect(() => {
// 当切换主题时,更新 html 当前的注入 class
updateGlobalThemeClass(getAppTheme.value)
// 注入全局宽高尺寸
setStyle(document.documentElement, {
'--html-height': `${height.value}px`,
'--html-width': `${width.value}px`,
})
})
expose()
},

View File

@ -1,6 +1,16 @@
import type { AppMenuConfig, PreloadingConfig } from '@/types'
import type { MessageProviderProps } from 'naive-ui'
/**
*
* @description
* html
*/
export const THEME_CLASS_NAMES = {
darkClassName: 'ray-template--dark',
lightClassName: 'ray-template--light',
}
/**
*
* @description

View File

@ -71,7 +71,9 @@ export const useAxiosInterceptor = () => {
func: AnyFC[],
fetchType: FetchType,
) => {
fetchType === 'ok' ? (implement[key] = func) : (errorImplement[key] = func)
fetchType === 'ok'
? (implement[key as keyof ImplementQueue] = func)
: (errorImplement[key as keyof ErrorImplementQueue] = func)
}
/** 获取队列中所有的所有拦截器方法 */
@ -79,7 +81,9 @@ export const useAxiosInterceptor = () => {
key: keyof ImplementQueue | keyof ErrorImplementQueue,
fetchType: FetchType,
): AnyFC[] => {
return fetchType === 'ok' ? implement[key] : errorImplement[key]
return fetchType === 'ok'
? implement[key as keyof ImplementQueue]
: errorImplement[key as keyof ErrorImplementQueue]
}
/** 队列执行器 */
@ -101,8 +105,8 @@ export const useAxiosInterceptor = () => {
) => {
const funcArr =
fetchType === 'ok'
? implement[implementKey]
: errorImplement[implementKey]
? implement[implementKey as keyof ImplementQueue]
: errorImplement[implementKey as keyof ErrorImplementQueue]
const instance = getAxiosInstance(key)
const { MODE } = getAppEnvironment()

View File

@ -0,0 +1,9 @@
import RFlow from './src/Flow'
import flowProps from './src/props'
import { useFlow } from './src/hooks'
import type { ExtractPublicPropTypes } from 'vue'
export type FlowProps = ExtractPublicPropTypes<typeof flowProps>
export { RFlow, flowProps, useFlow }

View File

@ -0,0 +1,155 @@
import './index.scss'
import '@logicflow/core/lib/style/index.css'
import { useTemplateRef } from 'vue'
import props from './props'
import { completeSize, call } from '@/utils'
import LogicFlow from '@logicflow/core'
import { omit } from 'lodash-es'
import type { FlowGraphData, G } from './types'
import type { WatchStopHandle } from 'vue'
// 是否首次注册插件
let isSetup = false
export default defineComponent({
name: 'RFlow',
props,
setup(props) {
// 流程图 dom 实例
const flowDomRef = useTemplateRef<HTMLElement>('flowDomRef')
// css 变量
const cssVars = computed(() => {
const { width, height } = props
const cssVar = {
'--r-flow-width': completeSize(width),
'--r-flow-height': completeSize(height),
}
return cssVar
})
// 流程图实例
const logicFlowInstRef = shallowRef<LogicFlow>()
// 需要禁用的流程图配置项
const readonlyOptions = {
nodeTextEdit: false,
edgeTextEdit: false,
textEdit: false,
}
// watchData 回调
let watchDataStop: WatchStopHandle
// 默认流程图数据
const defaultGraphData = {
nodes: [],
edges: [],
}
const cacheProps = {
readonly: props.readonly,
}
// 注册流程图插件
const registerExtension = () => {
if (!isSetup) {
props.use?.filter(Boolean).forEach((curr) => LogicFlow.use(curr))
isSetup = true
}
}
// 动态根据 readonly 配置项修改流程图配置
const updateFlowConfig = (bool: boolean) => {
if (!logicFlowInstRef.value) {
return
}
const ops = Object.entries(readonlyOptions).reduce(
(acc, [key]) => {
acc[key as keyof typeof acc] = !bool
return acc
},
{} as typeof readonlyOptions,
)
// 单独处理 isSilentMode 配置项
Object.assign(readonlyOptions, ops, {
isSilentMode: bool,
})
logicFlowInstRef.value.updateEditConfig(readonlyOptions)
}
/**
*
* @param graphData
*
* @description
*
*
* container 使
*/
const setupFlowRender = (graphData?: FlowGraphData) => {
registerExtension()
if (!flowDomRef.value) {
return
}
const { options, readonly } = props
// 初始化流程图实例
logicFlowInstRef.value = new LogicFlow({
container: unref(flowDomRef.value),
...omit(options, 'container'),
})
// 渲染
logicFlowInstRef.value.render((graphData || defaultGraphData) as G)
// 是否处于只读模式,如果是只读模式,则覆盖 options 配置项为 readonlyOptions
updateFlowConfig(readonly)
}
watchEffect(() => {
if (props.watchData) {
watchDataStop = watch(
() => props.data,
(ndata) => {
if (logicFlowInstRef.value) {
ndata && logicFlowInstRef.value.renderRawData(ndata as G)
} else {
setupFlowRender(ndata)
}
},
)
} else {
watchDataStop?.()
}
if (props.readonly !== cacheProps.readonly) {
updateFlowConfig(props.readonly)
cacheProps.readonly = props.readonly
}
})
onMounted(() => {
setupFlowRender()
const { onRegister } = props
if (onRegister && logicFlowInstRef.value) {
call(onRegister, logicFlowInstRef.value)
}
})
return {
flowDomRef,
cssVars,
}
},
render() {
const { cssVars } = this
return <div class="r-flow" style={[cssVars]} ref="flowDomRef"></div>
},
})

View File

@ -0,0 +1,17 @@
import type { FlowOptions } from './types'
/**
*
* @description
* RFlow
* 使 container使
*/
export const getDefaultFlowOptions = (): FlowOptions => {
return {
grid: true,
partial: false,
keyboard: {
enabled: true,
},
}
}

View File

@ -0,0 +1 @@
export { useFlow } from './useFlow'

View File

@ -0,0 +1,47 @@
import type LogicFlow from '@logicflow/core'
export const useFlow = () => {
let flowInst: LogicFlow
/**
*
* @param inst flow instance
*
* @description
* flow 使 useFlow hook
*/
const register = (inst: LogicFlow) => {
if (inst) {
flowInst = inst
}
}
/**
*
* @description
* flow
*
* onRegister
*
* @example
* const [register, { getFlowInstance }] = useFlow()
*
* const inst = getFlowInstance()
*/
const getFlowInstance = () => {
if (!flowInst) {
throw new Error(
'[useFlow]: flow instance is not ready yet. if you are using useFlow, please make sure you have called register method in onRegister event.',
)
}
return flowInst
}
return [
register,
{
getFlowInstance,
},
] as const
}

View File

@ -0,0 +1,10 @@
.r-flow {
width: var(--r-flow-width);
height: var(--r-flow-height);
}
.lf-text-input,
.lf-control-text,
.lf-menu-item {
color: initial;
}

View File

@ -0,0 +1,104 @@
import { getDefaultFlowOptions } from './constant'
import type { FlowGraphData, FlowOptions, ExtensionType } from './types'
import type LogicFlow from '@logicflow/core'
import type { MaybeArray } from '@/types'
const props = {
/**
*
* @description
*
*
*
* css
*
* @see https://site.logic-flow.cn/tutorial/extension/intro
*/
use: {
type: Array as PropType<ExtensionType[]>,
default: void 0,
},
/**
*
* @description
* data
*/
watchData: {
type: Boolean,
default: true,
},
/**
*
* @description
*
*
* options
*
* @default false
*/
readonly: {
type: Boolean,
default: false,
},
/**
*
* @description
*
*
* @default '100%'
*/
width: {
type: [String, Number] as PropType<string | number>,
default: '100%',
},
/**
*
* @description
*
*
* @default '100%'
*/
height: {
type: [String, Number] as PropType<string | number>,
default: '100%',
},
/**
*
* @description
*
*
* @default undefined
*/
data: {
type: Object as PropType<FlowGraphData>,
default: void 0,
},
/**
*
* @description
*
*
* @default undefined
*/
options: {
type: Object as PropType<FlowOptions>,
default: getDefaultFlowOptions(),
},
/**
*
* @description
* RFlow
* useFlow register 使便使 hooks
*
* @default undefined
*/
onRegister: {
type: [Function, Array] as PropType<
MaybeArray<(flowInst: LogicFlow) => void>
>,
default: void 0,
},
}
export default props

View File

@ -0,0 +1,61 @@
import type LogicFlow from '@logicflow/core'
import type { Recordable, SetRequired } from '@/types'
/**
*
* @description
* Omit
* Options
*
*
*/
type OptionsPickKeys =
| 'width'
| 'height'
| 'background'
| 'grid'
| 'partial'
| 'keyboard'
| 'style'
| 'edgeType'
| 'adjustEdge'
| 'textMode'
| 'edgeTextMode'
| 'nodeTextMode'
| 'allowRotate'
| 'allowResize'
| 'isSilentMode'
| 'stopScrollGraph'
| 'stopZoomGraph'
| 'stopMoveGraph'
| 'animation'
| 'history'
| 'outline'
| 'snapline'
| 'textEdit'
| 'guards'
| 'overlapMode'
| 'plugins'
| 'pluginsOptions'
| 'disabledPlugins'
| 'disabledTools'
| 'idGenerator'
| 'edgeGenerator'
| 'customTrajectory'
export type G = Parameters<LogicFlow['render']>[0]
export type NodeConfig = SetRequired<NonNullable<G['nodes']>[0], 'id'> &
Recordable
export type EdgeConfig = SetRequired<NonNullable<G['edges']>[0], 'id'> &
Recordable
export interface FlowGraphData {
nodes?: NodeConfig[]
edges?: EdgeConfig[]
}
export type FlowOptions = Pick<LogicFlow['options'], OptionsPickKeys>
export type ExtensionType = Parameters<typeof LogicFlow.use>[0]

View File

@ -5,12 +5,10 @@ import { R_MODAL_CLASS, CSS_VARS_KEYS } from '../constant'
import type { RModalProps } from '../types'
interface UseModalCreateOptions extends Omit<RModalProps, 'memo'> {}
const useModal = () => {
const { create: naiveCreate, destroyAll: naiveDestroyAll } = useNaiveModal()
const create = (options: UseModalCreateOptions) => {
const create = (options: RModalProps) => {
const { content, ...rest } = options
let contentNode = content
@ -27,7 +25,7 @@ const useModal = () => {
style: {
width: 'auto',
height:
'calc(100vh - 29px - var(--n-padding-bottom) - var(--n-padding-bottom) - var(--n-padding-top))',
'calc(var(--html-height) - 29px - var(--n-padding-bottom) - var(--n-padding-bottom) - var(--n-padding-top))',
},
},
{
@ -68,17 +66,6 @@ const useModal = () => {
// preset 为 cardfullscreen 为 true 时,最大化 modal
if (fullscreen && preset === 'card') {
const cardContentElement =
modalElement.querySelector<HTMLElement>('.n-card__content')
if (cardContentElement) {
setStyle(cardContentElement, {
maxHeight: `calc(100vh - 9px - var(--n-padding-bottom) - var(--n-padding-bottom) - var(--n-padding-top))`,
overflowY: 'hidden',
padding: '0',
})
}
setStyle(modalElement, {
width: '100%',
height: '100vh',

View File

@ -150,9 +150,9 @@ export default defineComponent({
const keys = Object.keys(propsPopselectValue.value)
keys.forEach((key) => {
propsPopselectValue.value[key] = value.includes(
key as PropsComponentPopselectKeys,
)
propsPopselectValue.value[
key as keyof typeof propsPopselectValue.value
] = value.includes(key as PropsComponentPopselectKeys)
})
}

View File

@ -191,8 +191,9 @@ export default defineComponent({
}
const fixedClick: FixedClick = (type, option, index) => {
const key = `${type}FixedActivated`
const otherKey = `${type === 'left' ? 'right' : 'left'}FixedActivated`
const key = `${type}FixedActivated` as const
const otherKey =
`${type === 'left' ? 'right' : 'left'}FixedActivated` as const
option[otherKey] = false
option[key] = !option[key]

View File

@ -13,6 +13,7 @@ export * from './base/RForm'
export * from './base/RSegment'
export * from './base/RBarcode'
export * from '../components/pro/RTablePro'
export * from './base/RFlow'
export { RCollapse }
// 导出自定义组件类型
@ -25,3 +26,9 @@ export type * from './base/RForm/src/types'
export type * from './base/RModal/src/types'
export type * from './base/RSegment/src/types'
export type * from './base/RBarcode/src/types'
export type {
NodeConfig,
EdgeConfig,
FlowGraphData,
FlowOptions,
} from './base/RFlow/src/types'

View File

@ -1,12 +1,12 @@
import type { Directive } from 'vue'
import type { App } from 'vue'
import type { Directive, App } from 'vue'
import type { Recordable } from '@/types'
export type { DebounceBindingOptions } from './modules/debounce/types'
export type { ThrottleBindingOptions } from './modules/throttle/types'
export type CustomDirectiveFC<T, K> = () => Directive<T, K>
export interface DirectiveModules<T = unknown, K = unknown> extends Object {
export interface DirectiveModules<T = unknown, K = unknown> extends Recordable {
default: CustomDirectiveFC<T, K>
}

View File

@ -11,7 +11,7 @@ export const combineDirective = <
const fc = directiveModules[curr]?.default
if (typeof fc === 'function') {
pre[curr] = fc
pre[curr as K] = fc
return pre
} else {

View File

@ -15,6 +15,8 @@
* createVariableState({ your state })
*/
import { updateObjectValue } from '@/utils'
import type { AnyFC } from '@/types'
/**
@ -57,11 +59,7 @@ export function setVariable<T extends VariableStateKey, FC extends AnyFC>(
value: VariableState[T],
cb?: FC,
) {
if (Object.hasOwn(variableState, key)) {
variableState[key] = value
cb?.()
}
updateObjectValue(variableState, key, value, cb)
}
/**

View File

@ -77,6 +77,7 @@ export function useBadge() {
/**
*
* @param target key AppMenuOption)
* @param extraOption
*
* @example
* const { show } = useBadge()
@ -84,8 +85,12 @@ export function useBadge() {
* show('your key')
* show({ ...AppMenuOption })
*/
const show = (target: BadgeKey) => {
const show = (
target: BadgeKey,
extraOption?: Omit<AppMenuExtraOptions, 'show'>,
) => {
normalOption(target, 'show', {
...extraOption,
show: true,
})
}

View File

@ -16,7 +16,7 @@ export const useWatermark = () => {
*/
const setWatermarkContent = (content: string) => {
const { getWatermarkConfig } = useSettingGetters()
const assignWatermark = Object.assign(getWatermarkConfig.value, {
const assignWatermark = Object.assign({}, getWatermarkConfig.value, {
content,
})
const { updateSettingState } = useSettingActions()

View File

@ -1,6 +1,8 @@
import dayjs from 'dayjs'
import { DEFAULT_DAYJS_LOCAL, DAYJS_LOCAL_MAP } from '@/app-config'
import type { DayjsLocalMap } from '@/types'
export interface FormatOption {
format?: string
}
@ -19,7 +21,7 @@ export interface StartAndEndOfDay {
formatEndOfDay: string
}
export type LocalKey = typeof DEFAULT_DAYJS_LOCAL
export type LocalKey = keyof DayjsLocalMap
const defaultDayjsFormat = 'YYYY-MM-DD HH:mm:ss'

View File

@ -133,7 +133,11 @@ export const useDomToImage = <T extends HTMLElement>(
return reject('useDomToImage: element is undefined.')
}
domToImageMethods[imageType ?? _imageType ?? 'jpeg']?.(element, options)
const imageTypeKey = (imageType ??
_imageType ??
'jpeg') as keyof typeof domToImageMethods
domToImageMethods[imageTypeKey]?.(element, options)
// eslint-disable-next-line @typescript-eslint/no-explicit-any
.then((res: any) => {
created?.(res, element)

View File

@ -1,8 +1,6 @@
import { unrefElement, effectDispose, isValueType, setStyle } from '@/utils'
import { useWindowSize } from '@vueuse/core'
import type { BasicTarget } from '@/types'
import type { CSSProperties } from 'vue'
export interface UseElementFullscreenOptions {
/**
@ -66,7 +64,6 @@ export interface UseElementFullscreenOptions {
let currentZIndex = 999
let isAppend = false
const ID_TAG = 'ELEMENT-FULLSCREEN-RAY'
const { width, height } = useWindowSize() // 获取实际高度避免 100vh 会导致手机端浏览器获取不准确问题
const styleElement = document.createElement('style')
/**
@ -140,8 +137,8 @@ export const useElementFullscreen = (
: zIndex,
'--element-fullscreen-transition': transition,
'--element-fullscreen-background-color': backgroundColor,
'--element-fullscreen-width': `${width.value}px`,
'--element-fullscreen-height': `${height.value}px`,
'--element-fullscreen-width': 'var(--html-width)',
'--element-fullscreen-height': 'var(--html-height)',
'--element-fullscreen-transform-x': `${catchBoundingClientRect.x}px`,
'--element-fullscreen-transform-y': `${catchBoundingClientRect.y}px`,
})
@ -217,8 +214,6 @@ export const useElementFullscreen = (
}
}
const stopWatch = watch(() => [width.value, height.value], updateStyle)
effectDispose(() => {
const element = unrefElement(target) as HTMLElement | null
@ -228,18 +223,12 @@ export const useElementFullscreen = (
// 回滚 z-index 值,避免无限增加
currentZIndex--
stopWatch()
})
return {
enter,
exit,
toggleFullscreen,
currentWindowSize: {
width,
height,
},
}
}

View File

@ -93,9 +93,9 @@ export default defineComponent({
<NTooltip placement="right">
{{
trigger: () => (
<h1 class="n-menu-item-content">
<NGradientText type="primary" size={18}>
{sideBarLogo.title?.[0] || null}
</h1>
</NGradientText>
),
default: () => sideBarLogo.title,
}}

View File

@ -80,15 +80,16 @@ export default defineComponent({
plain: true,
},
]
/** 初始化索引 */
// 初始化索引
let searchElementIndex = 0
/** 缓存索引 */
// 缓存索引
let preSearchElementIndex = searchElementIndex
const { isTabletOrSmaller } = useDevice()
const loading = ref(false)
const ACTIVE_CLASS = 'content-item--active' // 激活样式 class name
// 激活样式 class name
const ACTIVE_CLASS = 'content-item--active'
/** 初始化一些值 */
// 初始化一些值
const resetSearchSomeValue = () => {
state.searchOptions = []
state.searchValue = null
@ -96,7 +97,7 @@ export default defineComponent({
preSearchElementIndex = searchElementIndex
}
/** 按下 ctrl + k 或者 command + k 激活搜索栏 */
// 按下 ctrl + k 或者 command + k 激活搜索栏
const registerArouseKeyboard = (e: KeyboardEvent) => {
if (modelShow.value) {
return
@ -111,7 +112,14 @@ export default defineComponent({
}
}
/** 根据输入值模糊检索菜单 */
/**
*
* @param value
*
* @description
*
* getRoutes()
*/
const fuzzySearchMenuOptions = (value: string) => {
if (value) {
loading.value = true
@ -157,14 +165,15 @@ export default defineComponent({
})
loading.value = false
}, 500)
}, 300)
}
// 搜索结果项点击
const searchItemClick = (option: AppMenuOption) => {
if (option) {
const { meta } = option
/** 如果配置站外跳转则不会关闭搜索框 */
// 如果配置站外跳转则不会关闭搜索框
if (meta.windowOpen) {
window.open(meta.windowOpen)
} else {
@ -176,7 +185,7 @@ export default defineComponent({
}
}
/** 自动聚焦检索项 */
// 自动聚焦检索项
const autoFocusingSearchItem = () => {
const currentOption = state.searchOptions[searchElementIndex] // 获取当前搜索项
const preOption = state.searchOptions[preSearchElementIndex] // 获取上一搜索项
@ -205,7 +214,7 @@ export default defineComponent({
}
}
/** 渲染搜索菜单前缀图标, 如果没有则用 icon table 代替 */
// 渲染搜索菜单前缀图标,如果没有则用 icon table 代替
const RenderPreIcon = (meta: AppRouteMeta) => {
const { icon } = meta
@ -218,7 +227,7 @@ export default defineComponent({
}
}
/** 更新索引 */
// 更新索引
const updateIndex = (type: 'up' | 'down') => {
if (type === 'up') {
searchElementIndex -= 1
@ -237,7 +246,7 @@ export default defineComponent({
}
}
/** 注册按键: 上、下、回车 */
// 注册按键: 上、下、回车
const registerChangeSearchElementIndex = (e: KeyboardEvent) => {
const keyCode = e.key

View File

@ -71,6 +71,8 @@ export default defineComponent({
negativeText: '取消',
onPositiveClick: () => {
forIn(getDefaultSettingConfig(), (value, key) => {
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
modelReactive[key] = value
updateSettingState(key as keyof SettingState, value)

View File

@ -59,7 +59,9 @@ const avatarDropdownActionMap = {
},
}
export const avatarDropdownClick = (key: string | number) => {
export const avatarDropdownClick = (
key: keyof typeof avatarDropdownActionMap,
) => {
const action = avatarDropdownActionMap[key]
action ? action() : window.$message.info('这个人很懒, 没做这个功能~')

View File

@ -25,30 +25,26 @@
}
}
.r-layout-full__viewer-content--maximize--dark {
@include useAppTheme('dark') {
& .layout-content__maximize-out {
.ray-template--light {
.layout-content__maximize-out {
background-color: #d5d3d1;
color: #44403c;
&:hover {
color: #2c2a28;
background: #757473;
&:hover {
background-color: #d5d3d1;
color: #44403c;
}
}
}
}
.r-layout-full__viewer-content--maximize--light {
@include useAppTheme('light') {
& .layout-content__maximize-out {
.ray-template--dark {
.layout-content__maximize-out {
background-color: #44403c;
color: #d5d3d1;
&:hover {
color: #eae9e8;
background: #a19f9d;
&:hover {
background-color: #44403c;
color: #d5d3d1;
}
}
}
}

View File

@ -17,6 +17,12 @@
height: 100%;
}
}
// 设置 content-wrapper 的样式
.content-wrapper {
width: 100%;
height: 100%;
}
}
}
}

View File

@ -1,4 +1,4 @@
import { useElementBounding, useWindowSize } from '@vueuse/core'
import { useElementBounding } from '@vueuse/core'
import type { Ref } from 'vue'
@ -18,13 +18,12 @@ export const layoutCssVars = (
const siderBar = useElementBounding(element[0])
const menuTag = useElementBounding(element[1])
const footer = useElementBounding(element[2])
const { height, width } = useWindowSize()
return computed(() => {
return {
'--window-width': `${width.value}px`,
'--window-height': `${height.value}px`,
'--layout-content-height': `calc(${height.value}px - ${siderBar.height.value}px - ${menuTag.height.value}px - ${footer.height.value}px)`,
'--window-width': 'var(--html-width)',
'--window-height': 'var(--html-height)',
'--layout-content-height': `calc(var(--html-height) - ${siderBar.height.value}px - ${menuTag.height.value}px - ${footer.height.value}px)`,
'--layout-content-width': `${siderBar.width.value}px`,
'--layout-siderbar-height': `${siderBar.height.value}px`,
'--layout-menutag-height': `${menuTag.height.value}px`,

View File

@ -28,5 +28,6 @@
"Table": "Table",
"TemplateHooks": "Template Api",
"scrollReveal": "Scroll Reveal",
"TablePro": "Table Pro"
"TablePro": "Table Pro",
"Flow": "Flow"
}

View File

@ -28,5 +28,6 @@
"Table": "表格",
"TemplateHooks": "模板内置 Api",
"scrollReveal": "滚动动画",
"TablePro": "高级表格"
"TablePro": "高级表格",
"Flow": "流程图"
}

View File

@ -21,7 +21,7 @@ export const getAppLocalMessages = async (
for (const curr of localOptions) {
const msg: AppLocalesModules = await import(`@/locales/lang/${curr.key}.ts`)
const key = curr.key
const key = curr.key as keyof AppCurrentAppMessages
if (key) {
message[key] = msg?.default?.message ?? {}

View File

@ -0,0 +1,16 @@
import { t } from '@/hooks/web/useI18n'
import { LAYOUT } from '@/router/constant'
import type { AppRouteRecordRaw } from '@/router/types'
const r: AppRouteRecordRaw = {
path: '/flow',
component: () => import('@/views/demo/Flow'),
meta: {
i18nKey: t('menu.Flow'),
icon: 'other',
order: 2,
},
}
export default r

View File

@ -121,14 +121,14 @@ export const piniaMenuStore = defineStore(
* fullPath
*
* @example
* resolveOption({ path: '/dashboard', name: 'Dashboard', meta: { i18nKey: 'menu.Dashboard' } })
* resolveOption({ path: '/demo', fullPath: '/demo', name: 'Demo', meta: { ... } })
* resolveOption({ ...VueRouterRouteOption })
*/
const resolveOption = (option: AppMenuOption) => {
const { meta } = option
const { i18nKey, noLocalTitle } = meta
/** 设置 label, i18nKey 优先级最高 */
// 设置 label, i18nKey 优先级最高
const label = computed(() => (i18nKey ? t(`${i18nKey}`) : noLocalTitle))
/**
*
@ -144,7 +144,7 @@ export const piniaMenuStore = defineStore(
}),
breadcrumbLabel: label.value,
} as AppMenuOption
/** 合并 icon, extra */
// 合并 icon, extra
const attr: AppMenuOption = Object.assign({}, route, {
icon: createMenuIcon(option),
extra: createMenuExtra(option),
@ -161,9 +161,12 @@ export const piniaMenuStore = defineStore(
/**
*
*
* @param key menu state key
*
* ,
* @description
*
*
*
*/
const setBreadcrumbOptions = (key: string | number) => {
menuState.breadcrumbOptions = parseAndFindMatchingNodes(
@ -175,8 +178,12 @@ export const piniaMenuStore = defineStore(
/**
*
* @param options menu tag option(s)
* @param isAppend true: (push), false:
* @param options menu tag options
* @param isAppend is append
*
* @description
*
* true: pushfalse:
*/
const setMenuTagOptions = (
options: MenuTagOptions | MenuTagOptions[],
@ -190,7 +197,14 @@ export const piniaMenuStore = defineStore(
: (menuState.menuTagOptions = arr)
}
/** 当 url 地址发生变化触发 menuTagOptions 更新 */
/**
*
* @param key full path
* @param option menu tag option
*
* @description
*
*/
const setMenuTagOptionsWhenMenuValueChange = (
key: string | number,
option: AppMenuOption,
@ -324,8 +338,11 @@ export const piniaMenuStore = defineStore(
/**
*
*
* url fullPath
* @description
*
* url fullPath
*
*
*/
const setupAppMenu = () => {
return new Promise<void>((resolve) => {
@ -384,6 +401,9 @@ export const piniaMenuStore = defineStore(
/**
*
* @param collapsed
*
* @description
*
*/
const collapsedMenu = (collapsed: boolean) =>
(menuState.collapsed = collapsed)
@ -394,14 +414,18 @@ export const piniaMenuStore = defineStore(
* @param length
*
* @returns
*
* @description
* menu tag
*/
const spliceMenTagOptions = (idx: number, length = 1) =>
menuState.menuTagOptions.splice(idx, length)
/**
*
*
*
* @description
*
*
*/
const setupPiniaMenuStore = async () => {
if (!isSetupAppMenuLock.value) {
@ -413,7 +437,7 @@ export const piniaMenuStore = defineStore(
isSetupAppMenuLock.value = false
}
/** 监听路由变化并且更新路由菜单与菜单标签 */
// 监听路由变化并且更新路由菜单与菜单标签
watch(
() => route.fullPath,
async (ndata, odata) => {

View File

@ -30,7 +30,7 @@ const isMatch = (
return false
}
return node[key] === value
return node[key as keyof AppMenuOption] === value
}
/**

View File

@ -120,11 +120,11 @@ export const piniaSettingStore = defineStore(
settingState.primaryColorOverride.common = themeOverrides
const body = document.body
const html = document.documentElement
/** 设置主题色变量 */
body.style.setProperty('--ray-theme-primary-color', value)
body.style.setProperty('--ray-theme-primary-fade-color', alphaColor)
html.style.setProperty('--ray-theme-primary-color', value)
html.style.setProperty('--ray-theme-primary-fade-color', alphaColor)
}
/**

View File

@ -93,7 +93,7 @@ body {
}
// 配合 v-disabled 指令使用
body .ray-template__directive--disabled {
html .ray-template__directive--disabled {
opacity: 0.3 !important;
pointer-events: none !important;
cursor: not-allowed !important;

View File

@ -43,7 +43,7 @@
// 根据主题切换样式
@mixin useAppTheme($theme) {
body[class="ray-template--#{$theme}"] & {
html[class='ray-template--#{$theme}'] & {
@content;
}
}

View File

@ -56,7 +56,7 @@ export declare global {
bus: EventBus
shadowRoot?: ShadowRoot
props?: { [key: string]: unknown }
location?: Object
location?: object
}
$message: MessageApi

View File

@ -113,3 +113,14 @@ export type DeepReadonly<T> = T extends object
readonly [P in keyof T]: DeepReadonly<T[P]>
}
: T
/**
*
* @description
*
*
* @example
* SetRequired<{ a: string, b?: number }, 'a'> // { a: string, b: number }
*/
export type SetRequired<T, K extends keyof T> = Omit<T, K> &
Required<Pick<T, K>>

View File

@ -62,6 +62,7 @@ export type BasicTypes =
| string
| symbol
| bigint
// eslint-disable-next-line @typescript-eslint/no-unsafe-function-type
| Function
| any[]
| object

View File

@ -5,6 +5,7 @@ import type {
BasicTarget,
QueryElementsOptions,
ElementSelector,
Recordable,
} from '@/types'
import type { CSSProperties } from 'vue'
@ -160,7 +161,7 @@ export const hasClass = (
*/
export const autoPrefixStyle = (style: string) => {
const prefixes = ['webkit', 'moz', 'ms', 'o']
const styleWithPrefixes = {}
const styleWithPrefixes: Recordable = {}
prefixes.forEach((prefix) => {
styleWithPrefixes[
@ -214,16 +215,17 @@ export const setStyle = <Style extends CSSProperties>(
if (key.startsWith('--')) {
element.style.setProperty(trimKey, trimValue)
} else if (key.startsWith('-')) {
element.style[key] = value
element.style.setProperty(key, value)
} else {
// 兼容浏览器前缀
const kitFix = autoPrefixStyle(trimKey)
Object.keys(kitFix).forEach((key) => {
element.style[key] = kitFix[key]
element.style.setProperty(key, kitFix[key])
})
// 设置默认需要添加样式
element.style[trimKey] = trimValue
element.style.setProperty(trimKey, trimValue)
}
}
})
@ -246,7 +248,7 @@ export const setStyle = <Style extends CSSProperties>(
const keys = Object.keys(styles)
keys.forEach((curr) => {
set(`${curr}: ${styles[curr]}`, element)
set(`${curr}: ${styles[curr as keyof typeof styles]}`, element)
})
}
}

View File

@ -31,7 +31,7 @@ export interface RenderNodeOptions<T extends DefaultElement> {
* renderNode('hello world') // () => 'hello world'
* renderNode(<div>hello world</div>) // () => <div>hello world</div>
* renderNode(() => <div>hello world</div>) // () => <div>hello world</div>
* renderNode(null, { defaultElement: () => <span>hello world</span> }) // () => 'hello world'
* renderNode(null, { defaultElement: () => <span>hello world</span> }) // () => <span>hello world</span>
*/
export function renderNode<T extends DefaultElement>(
vnode: RenderVNodeType,

87
src/views/demo/Flow.tsx Normal file
View File

@ -0,0 +1,87 @@
import { RFlow, RForm } from '@/components'
import { useFlow } from '@/components'
import type { FlowGraphData } from '@/components'
import { NCard, NFlex, NFormItemGridItem, NGrid, NSwitch } from 'naive-ui'
export default defineComponent({
name: 'RFlowDemo',
setup() {
const [register, { getFlowInstance }] = useFlow()
const flowDataRef = ref<FlowGraphData>()
const settingRef = ref({
readonly: false,
})
const getInst = () => {
console.log(getFlowInstance())
}
setTimeout(() => {
flowDataRef.value = {
// 节点
nodes: [
{
id: '21',
type: 'rect',
x: 300,
y: 100,
text: 'rect node',
},
{
id: '50',
type: 'circle',
x: 500,
y: 100,
text: 'circle node',
},
],
// 边
edges: [
{
id: '21',
type: 'polyline',
sourceNodeId: '50',
targetNodeId: '21',
},
],
}
}, 1000)
onMounted(() => {
getInst()
})
return {
register,
flowDataRef,
settingRef,
}
},
render() {
const { register, flowDataRef, settingRef } = this
return (
<NFlex vertical>
<NCard>
<RForm>
<NGrid xGap={4} yGap={18} cols={4}>
<NFormItemGridItem label="禁用流程图">
<NSwitch v-model:value={settingRef.readonly} />
</NFormItemGridItem>
</NGrid>
</RForm>
</NCard>
<NCard>
<RFlow
height={500}
onRegister={register}
data={flowDataRef}
readonly={settingRef.readonly}
/>
</NCard>
</NFlex>
)
},
})

View File

@ -18,7 +18,9 @@ export default defineComponent({
title: '卡片模态框',
dad: true,
preset: 'card',
content: '我可以被拖拽的全屏card模态框',
content: () => (
<div style="height: 3000px;">card模态框</div>
),
fullscreen: true,
})
}
@ -59,7 +61,7 @@ export default defineComponent({
fullscreen
preset="card"
>
<div style="height: 3000px;">card模态框</div>
</RModal>
<RModal
v-model:show={this.modal2}

View File

@ -57,7 +57,7 @@ export default defineComponent({
Object.keys(obj).reduce((pre, curr) => {
pre.push({
name: curr,
relyVersion: obj[curr],
relyVersion: obj[curr as keyof typeof obj],
relyAddress: '',
})

View File

@ -99,7 +99,13 @@ export default defineComponent({
<NButton onClick={() => badgeHidden('/template-hooks')}>
</NButton>
<NButton onClick={() => badgeShow('/template-hooks')}>
<NButton
onClick={() =>
badgeShow('/template-hooks', {
label: this.badgeValue,
})
}
>
</NButton>
<NButton

View File

@ -26,9 +26,7 @@
"@mock/*": ["mock/*"],
"@mock": ["mock/*"]
},
"suppressImplicitAnyIndexErrors": true,
"types": ["vite/client", "vitest/globals"],
"ignoreDeprecations": "5.0"
"types": ["vite/client", "vitest/globals"]
},
"include": [
"vite.config.ts",

View File

@ -17,7 +17,7 @@ import viteVeI18nPlugin from '@intlify/unplugin-vue-i18n/vite'
import viteInspect from 'vite-plugin-inspect'
import viteSvgLoader from 'vite-svg-loader'
import { analyzer, adapter } from 'vite-bundle-analyzer'
import viteCompression from 'vite-plugin-compression'
import gzipPlugin from 'rollup-plugin-gzip'
import { ViteEjsPlugin as viteEjsPlugin } from 'vite-plugin-ejs'
import viteAutoImport from 'unplugin-auto-import/vite'
import viteEslint from 'vite-plugin-eslint'
@ -180,7 +180,7 @@ function baseOptions(mode: string): PluginOption[] {
},
],
}),
viteCompression(),
gzipPlugin(),
viteSvgLoader({
defaultImport: 'url', // 默认以 url 形式导入 svg
}),