This commit is contained in:
XiaoDaiGua-Ray 2023-10-27 14:55:25 +08:00
parent f6c343911e
commit a840693455
33 changed files with 361 additions and 184 deletions

View File

@ -1,5 +1,36 @@
# CHANGE LOG
## 4.2.8
我好像犯了一个很愚蠢的错误,那就是使用 useFullscreen 方法的时候总是会弹出提示。所以紧急修复了这个很愚蠢的问题,并且移除了这个方法。
在兼容移动端后,手机打开页面会触发缩放情况,这个版本禁用了缩放。如果有需求可以去设置 `index.html` 中的 `meta` 标签。并且优化了移动端的锁屏样式。
重构了 RChart 组件,因为该组件过于单调,所以决定赋予更多的功能。结合 naive NCard 组件进行了二次封装。当然你也可以通过设置 `preset = default | null | void 0` 调用无预设样式的 chart 图表。
### Feats
- 移除二次封装 useFullscreen 方法
- 禁用自动缩放
- 优化移动端锁屏样式
- 禁用 vue-i18n 自动导入
- RChart
- 新增 preset 属性配置,以下配置属性仅在 preset 为 card 生效
- 新增 downloadOptions 下载图片配置具体说明查看文档https://echarts.apache.org/zh/api.html#echartsInstance.getDataURL
- 修改 RChart 渲染时机,更改为主进程渲染
- 新增 cardExtra slots 自定义配置操作栏
- 新增 title 属性
- 新增 onDropdownSelect 回调方法
- 新增 renderNode 方法,用于减少 if else 表达式渲染 vnode
- 语言包新增 globalMessage 模块,用于配置统一的输出消息
### Fixes
- 修复每次执行 useFullscreen 都弹出提示的问题
- RChart
- 修复 animation false 状态渲染异常问题
- 修复响应式代理 echart instance 时,导致部分方法异常问题
## 4.2.7
主要是做了一些统一命名的事情,以前由于写的比较放浪形骸现在正在慢慢更改这个大问题。

View File

@ -3,7 +3,10 @@
<head>
<meta charset="UTF-8" />
<link rel="icon" type="image/svg+xml" href="/ray.svg" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<meta
name="viewport"
content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"
/>
<title>Vite + Vue + TS</title>
</head>
<style>

View File

@ -1,7 +1,7 @@
{
"name": "ray-template",
"private": false,
"version": "4.2.7",
"version": "4.2.8",
"type": "module",
"engines": {
"node": ">=16.0.0",
@ -53,7 +53,6 @@
"@types/crypto-js": "^4.1.1",
"@types/lodash-es": "^4.17.7",
"@types/mockjs": "1.0.7",
"@types/scrollreveal": "^0.0.8",
"@typescript-eslint/eslint-plugin": "^5.61.0",
"@typescript-eslint/parser": "^5.61.0",
"@vitejs/plugin-vue": "^4.2.3",

View File

@ -75,6 +75,7 @@ const LockScreen = defineComponent({
type="password"
placeholder="请输入锁屏密码"
clearable
showPasswordOn="click"
minlength={6}
maxlength={12}
onKeydown={(e: KeyboardEvent) => {

View File

@ -18,6 +18,7 @@ import dayjs from 'dayjs'
import { useSetting, useSignin } from '@/store'
import { rules, useCondition } from '@/app-components/app/AppLockScreen/hook'
import useAppLockScreen from '@/app-components/app/AppLockScreen/appLockVar'
import { useDevice } from '@/hooks/web/index'
import type { FormInst, InputInst } from 'naive-ui'
@ -30,6 +31,7 @@ const UnlockScreen = defineComponent({
const { logout } = useSignin()
const { changeSwitcher } = useSetting()
const { setLockAppScreen } = useAppLockScreen()
const { isTabletOrSmaller } = useDevice()
const HH_MM_FORMAT = 'HH:mm'
const AM_PM_FORMAT = 'A'
@ -91,13 +93,23 @@ const UnlockScreen = defineComponent({
unlockScreen,
formRef,
inputInstRef,
isTabletOrSmaller,
}
},
render() {
const { isTabletOrSmaller } = this
return (
<div class="app-lock-screen__unlock">
<div class="app-lock-screen__unlock__content">
<div class="app-lock-screen__unlock__content-bg">
<div
class={[
'app-lock-screen__unlock__content-bg',
isTabletOrSmaller
? 'app-lock-screen__unlock__content-bg--smaller'
: '',
]}
>
<div class="left">{this.HH_MM?.split(':')[0]}</div>
<div class="right">{this.HH_MM?.split(':')[1]}</div>
</div>

View File

@ -26,6 +26,16 @@
gap: 80px;
z-index: 0;
&.app-lock-screen__unlock__content-bg--smaller {
& .left,
& .right {
padding: 0px;
font-size: 90px;
padding: 24px;
border-radius: 4px;
}
}
& .left,
& .right {
@include flexCenter;

View File

@ -43,6 +43,7 @@ const AppLockScreen = defineComponent({
v-model:show={this.lockScreenSwitch}
transformOrigin="center"
show
autoFocus={false}
maskClosable={false}
closeOnEsc={false}
preset={!this.getLockAppScreen() ? 'dialog' : void 0}

View File

@ -37,24 +37,27 @@ import {
} from 'echarts/charts' // 系列类型(后缀都为 `SeriesOption`)
import { LabelLayout, UniversalTransition } from 'echarts/features' // 标签自动布局, 全局过渡动画等特性
import { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器
import { NCard } from 'naive-ui'
import props from './props'
import { useSetting } from '@/store'
import { cloneDeep, throttle } from 'lodash-es'
import { on, off, completeSize } from '@/utils/element'
import { throttle } from 'lodash-es'
import { completeSize } from '@/utils/element'
import { call } from '@/utils/vue/index'
import { setupChartTheme } from './helper'
import { APP_THEME } from '@/app-config/designConfig'
import { useResizeObserver } from '@vueuse/core'
import RMoreDropdown from '@/components/RMoreDropdown/index'
import { renderNode } from '@use-utils/vue/index'
import { downloadBase64File } from '@use-utils/basic'
import type { WatchStopHandle } from 'vue'
import type { AnyFC } from '@/types/modules/utils'
import type { DebouncedFunc } from 'lodash-es'
import type { ChartTheme } from '@/components/RChart/type'
import type { UseResizeObserverReturn } from '@vueuse/core'
import type { ECharts, EChartsCoreOption } from 'echarts/core'
export type { RayChartInst, EChartsExtensionInstallRegisters } from './type'
import type { RenderVNodeType } from '@use-utils/vue/renderNode'
import type { DropdownProps, DropdownOption } from 'naive-ui'
const defaultChartOptions = {
notMerge: false,
@ -70,14 +73,24 @@ export default defineComponent({
setup(props, { expose }) {
const settingStore = useSetting()
const { themeValue: currentTheme } = storeToRefs(settingStore)
const rayChartRef = ref<HTMLElement>() // `echart` 容器实例
const rayChartRef = ref<HTMLElement>() // echart 容器实例
const rayChartWrapperRef = ref<HTMLElement>()
const echartInstanceRef = ref<ECharts>() // `echart` 实例
const echartInstanceRef = ref<ECharts>() // echart 实例
let resizeThrottleReturn: DebouncedFunc<AnyFC> | null // resize 防抖方法实例
let resizeOvserverReturn: UseResizeObserverReturn | null
const { echartTheme } = APP_THEME
let watchCallback: WatchStopHandle | null
let echartInst: ECharts | null // 无代理响应式代理缓存 echart inst
const moreDropDownOptions = computed<DropdownProps['options']>(() => [
{
label: '下载图片',
key: 'downloadChart',
disabled:
echartInstanceRef.value && echartInstanceRef.value.getDom()
? false
: true,
},
])
const cssVarsRef = computed(() => {
const cssVars = {
'--ray-chart-width': completeSize(props.width),
@ -105,7 +118,6 @@ export default defineComponent({
ToolboxComponent,
AriaComponent,
]) // 注册组件
echarts.use([
BarChart,
LineChart,
@ -114,11 +126,7 @@ export default defineComponent({
ScatterChart,
PictorialBarChart,
]) // 注册 chart series type
echarts.use([LabelLayout, UniversalTransition]) // 注册布局, 过度效果
// 如果业务场景中需要 `svg` 渲染器, 手动导入渲染器后使用该行代码即可(不过为了体积考虑, 移除了 SVG 渲染器)
// echarts.use([props.canvasRender ? CanvasRenderer : SVGRenderer])
echarts.use([CanvasRenderer]) // 注册渲染器
try {
@ -132,7 +140,13 @@ export default defineComponent({
/**
*
* chart
* chart
*
* theme autoChangeTheme
* theme default chart
*
* Boolean(theme) false echartTheme
* echartTheme 使
*/
const updateChartTheme = () => {
if (props.theme === 'default') {
@ -159,11 +173,10 @@ export default defineComponent({
* @returns `chart options`
*
*
*
* ...
*/
const combineChartOptions = (ops: EChartsCoreOption) => {
let options = cloneDeep(unref(ops))
let options = unref(ops)
const assign = (opts: object) =>
Object.assign(
@ -211,25 +224,28 @@ export default defineComponent({
})
/** 注册 chart */
echartInstanceRef.value = echarts.init(element, theme, {
echartInst = echarts.init(element, theme, {
/** 如果款度为 0, 则以 200px 填充 */
width: width === 0 ? 200 : void 0,
/** 如果高度为 0, 则以 200px 填充 */
height: height === 0 ? 200 : void 0,
})
echartInstanceRef.value = echartInst
/** 设置 options 配置项 */
echartInstanceRef.value.setOption({})
if (props.animation) {
echartInst.setOption({})
setTimeout(() => {
options && echartInstanceRef.value?.setOption(options)
options && echartInst?.setOption(options)
})
} else {
options && echartInst?.setOption(options)
}
/** 渲染成功回调 */
if (onSuccess) {
call(onSuccess, echartInstanceRef.value)
call(onSuccess, echartInst)
}
} catch (e) {
/** 渲染失败回调 */
@ -246,25 +262,40 @@ export default defineComponent({
* `chart` ,
*/
const destroyChart = () => {
if (echartInstanceRef.value) {
echartInstanceRef.value.clear()
echartInstanceRef.value.dispose()
if (echartInst && echartInst.getDom()) {
echartInst.clear()
echartInst.dispose()
echartInstanceRef.value = void 0
}
}
/** 重置 echarts 尺寸 */
const resizeChart = () => {
if (echartInstanceRef.value) {
try {
echartInstanceRef.value.resize()
// eslint-disable-next-line no-empty
} catch (e) {}
if (echartInst) {
echartInst.resize()
}
}
const dropdownSelect = (key: string | number, option: DropdownOption) => {
if (key === 'downloadChart' && echartInst && echartInst.getDom()) {
const { filename, ...args } = props.downloadOptions
downloadBase64File(
echartInst.getDataURL(args),
filename ?? `${new Date().getTime()}`,
)
}
const { onDropdownSelect } = props
if (onDropdownSelect) {
call(onDropdownSelect, key, option)
}
}
const mount = () => {
// 避免重复渲染
if (echartInstanceRef.value?.getDom()) {
if (echartInst?.getDom()) {
console.warn(
'RChart mount: There is a chart instance already initialized on the dom. Execution was interrupted',
)
@ -278,20 +309,17 @@ export default defineComponent({
if (props.autoResize) {
resizeThrottleReturn = throttle(resizeChart, props.throttleWait)
/** 监听内容区域尺寸变化更新 chart */
resizeOvserverReturn = useResizeObserver(
props.observer || rayChartWrapperRef,
resizeThrottleReturn,
)
on(window, 'resize', resizeThrottleReturn)
}
}
const unmount = () => {
/** 卸载 echarts */
destroyChart()
/** 卸载事件柄 */
resizeThrottleReturn && off(window, 'resize', resizeThrottleReturn)
/** 注销防抖 */
resizeThrottleReturn?.cancel()
/** 注销 observer 监听 */
@ -342,7 +370,7 @@ export default defineComponent({
defaultChartOptions,
)
/** 如果 options 发生变动更新 echarts */
echartInstanceRef.value?.setOption(options, setOpt)
echartInst?.setOption(options, setOpt)
},
{
deep: true,
@ -353,8 +381,8 @@ export default defineComponent({
}
props.loading
? echartInstanceRef.value?.showLoading(props.loadingOptions)
: echartInstanceRef.value?.hideLoading()
? echartInst?.showLoading(props.loadingOptions)
: echartInst?.hideLoading()
})
expose({
@ -368,9 +396,7 @@ export default defineComponent({
await registerChartCore()
})
onMounted(() => {
nextTick(() => {
mount()
})
mount()
})
onBeforeUnmount(() => {
unmount()
@ -381,10 +407,51 @@ export default defineComponent({
rayChartRef,
cssVarsRef,
rayChartWrapperRef,
moreDropDownOptions,
dropdownSelect,
}
},
render() {
return (
const {
title,
contentStyle,
preset,
moreDropDownOptions,
dropdownSelect,
bordered,
dropdownOptions,
} = this
const { cardExtra } = this.$slots
return preset === 'card' ? (
<NCard
class="ray-chart"
ref="rayChartWrapperRef"
style={[this.cssVarsRef]}
contentStyle={contentStyle}
bordered={bordered}
>
{{
default: () => (
<div class="ray-chart__container" ref="rayChartRef"></div>
),
header: renderNode(title, {
defaultElement: <div style="display: none;"></div>,
}),
'header-extra': renderNode(cardExtra as RenderVNodeType, {
defaultElement: (
<RMoreDropdown
iconSize={18}
cursor="pointer"
options={dropdownOptions ?? moreDropDownOptions}
trigger="click"
onSelect={dropdownSelect.bind(this)}
/>
),
}),
}}
</NCard>
) : (
<div class="ray-chart" style={[this.cssVarsRef]} ref="rayChartWrapperRef">
<div class="ray-chart__container" ref="rayChartRef"></div>
</div>

View File

@ -1,5 +1,5 @@
import type * as echarts from 'echarts/core' // `echarts` 核心模块
import type { PropType } from 'vue'
import type { PropType, VNode } from 'vue'
import type { MaybeArray } from '@/types/modules/utils'
import type {
LoadingOptions,
@ -8,11 +8,60 @@ import type {
} from '@/components/RChart/type'
import type { ECharts, SetOptionOpts } from 'echarts/core'
import type { MaybeComputedElementRef, MaybeElement } from '@vueuse/core'
import type { EChartsExtensionInstallRegisters } from './type'
import type {
EChartsExtensionInstallRegisters,
RChartPresetType,
RChartDownloadOptions,
} from './type'
import type { CardProps, DropdownProps, DropdownOption } from 'naive-ui'
import { loadingOptions } from './helper'
const props = {
bordered: {
/**
*
* preset card
*
*
*/
type: Boolean,
default: true,
},
downloadOptions: {
/**
*
* preset card
*
* type: png, jpg, svgpng, jpg canvas 使svg 使 svg
* pixelRatio: 导出的图片分辨率比例 1
* backgroundColor: 导出的图片背景色使 option backgroundColor
* excludeComponents: 忽略组件的列表 toolbox ['toolbox']
*/
type: Object as PropType<RChartDownloadOptions>,
default: () => ({}),
},
onDropdownSelect: {
// 仅在 preset 为 card 时生效
type: [Function, Array] as PropType<
MaybeArray<(key: string | number, option: DropdownOption) => void>
>,
},
dropdownOptions: {
// 仅在 preset 为 card 时生效
type: Array as PropType<DropdownProps['options']>,
},
preset: {
type: String as PropType<RChartPresetType>,
},
contentStyle: {
// 仅在 preset 为 card 时生效
type: [String, Object] as PropType<CardProps['contentStyle']>,
},
title: {
// 仅在 preset 为 card 时生效
type: [String, Function] as PropType<string | (() => VNode)>,
},
width: {
/**
*
@ -44,17 +93,6 @@ const props = {
type: [Boolean, Object] as PropType<AutoResize>,
default: true,
},
canvasRender: {
/**
*
* @deprecated
* `chart` , 使 `canvas`
*
* , `SVGRenderer`
*/
type: Boolean,
default: true,
},
showAria: {
/**
*

View File

@ -63,14 +63,24 @@ export interface RayChartInst {
*
* chart
* chart
*
* @default () => void
*/
dispose: () => void
/**
*
* chart
* options props chart
*
* @default () => void
*/
render: () => void
}
export type EChartsExtensionInstallRegisters = typeof CanvasRenderer
export type RChartPresetType = 'card' | 'default' | null | undefined
export type RChartDownloadOptions = {
filename?: string
} & Parameters<ECharts['getDataURL']>[0]

View File

@ -18,14 +18,14 @@ export default defineComponent({
name: 'RMoreDropdown',
props,
render() {
const { iconSize } = this
const { iconSize, cursor } = this
return (
<NDropdown {...this.$props} {...this.$attrs}>
{this.$slots.default ? (
this.$slots.default()
) : (
<RIcon name="more" size={iconSize} />
<RIcon name="more" size={iconSize} cursor={cursor} />
)}
</NDropdown>
)

View File

@ -17,6 +17,10 @@ const props = {
type: Number,
default: 14,
},
cursor: {
type: String,
default: 'default',
},
}
export default props

View File

@ -18,7 +18,7 @@ import C from './components/C'
import Print from './components/Print'
import props from './props'
import { call } from '@/utils/vue/index'
import { call, renderNode } from '@/utils/vue/index'
import { uuid } from '@/utils/basic'
import config from './config'
@ -203,7 +203,9 @@ export default defineComponent({
) : null}
</>
),
header: () => this.title || <div style="display: none;"></div>,
header: renderNode(this.title, {
defaultElement: <div style="display: none;"></div>,
}),
'header-extra': () => (
<NSpace wrapItem={false} align="center">
{tool(this.$props as any)}

View File

@ -167,7 +167,6 @@ export default defineComponent({
}
const resizableClick = (option: C, index: number) => {
console.log('🚀 ~ resizableClick ~ option:', option.isResizable)
option['isResizable'] = !option['isResizable']
option['resizable'] = option['isResizable']
treeDataSource.value[index] = option

View File

@ -13,7 +13,7 @@ import { NPopover } from 'naive-ui'
import RIcon from '@/components/RIcon/index'
import config from '../config'
import { useFullscreen } from '@/hooks/web/index'
import { useFullscreen } from 'vue-hooks-plus'
import type { TableProvider } from '../type'
@ -24,14 +24,18 @@ export default defineComponent({
config.tableKey,
{} as TableProvider,
)
const [isFullscreen, { toggleFullscreen }] = useFullscreen(wrapperRef)
const [isFullscreen, { toggleFullscreen, isEnabled }] =
useFullscreen(wrapperRef)
return {
toggleFullscreen,
isFullscreen,
isEnabled,
}
},
render() {
const { toggleFullscreen, isEnabled, $t } = this
return (
<NPopover showArrow={false}>
{{
@ -40,7 +44,13 @@ export default defineComponent({
name="fullscreen"
size={config.tableIconSize}
cursor="pointer"
onClick={this.toggleFullscreen.bind(this)}
onClick={() => {
if (!isEnabled) {
$t('globalMessage.isEnabledFullscreen')
}
toggleFullscreen()
}}
/>
),
default: () => (this.isFullscreen ? '取消全屏' : '全屏表格'),

View File

@ -9,7 +9,7 @@
"#d2f5a6",
"#76f2f2"
],
"backgroundColor": "rgba(41,52,65,1)",
"backgroundColor": "#18181c",
"textStyle": {},
"title": {
"textStyle": {

View File

@ -37,5 +37,5 @@ export function getVariable(key: VariableStateKey) {
}
export function globalVariableToRefs<K extends VariableStateKey>(key: K) {
return toRef<typeof variableState, K>(variableState, key)
return readonly(toRef<typeof variableState, K>(variableState, key))
}

View File

@ -13,6 +13,5 @@ import { useI18n, t } from './useI18n'
import { useVueRouter } from '../web/useVueRouter'
import { useDayjs } from '../web/useDayjs'
import { useDevice } from './useDevice'
import { useFullscreen } from './useFullscreen'
export { useI18n, useVueRouter, useDayjs, t, useDevice, useFullscreen }
export { useI18n, useVueRouter, useDayjs, t, useDevice }

View File

@ -1,40 +0,0 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-10-25
*
* @workspace ray-template
*
* @remark
*/
import { useFullscreen as hooksPlusUseFullscreen } from 'vue-hooks-plus'
import type { useFullscreen as UseFullscreen } from 'vue-hooks-plus'
type UseFullscreenParams = Parameters<typeof UseFullscreen>
export function useFullscreen(
target: UseFullscreenParams[0],
options?: UseFullscreenParams[1],
) {
const [
isFullscreen,
{ enterFullscreen, exitFullscreen, toggleFullscreen, isEnabled },
] = hooksPlusUseFullscreen(target, options)
if (!isEnabled) {
window.$message.warning('您当前环境不支持全屏模式')
}
return [
isFullscreen,
{
enterFullscreen,
exitFullscreen,
toggleFullscreen,
isEnabled,
},
] as const
}

View File

@ -13,12 +13,10 @@ import { NSpace, NSwitch, NTooltip } from 'naive-ui'
import RIcon from '@/components/RIcon'
import { useSetting } from '@/store'
import { useI18n } from '@/hooks/web/index'
const ThemeSwitch = defineComponent({
name: 'ThemeSwitch',
setup() {
const { t } = useI18n()
const settingStore = useSetting()
const { changeSwitcher } = settingStore
const { themeValue } = storeToRefs(settingStore)
@ -34,14 +32,13 @@ const ThemeSwitch = defineComponent({
}
return {
t,
changeSwitcher,
themeValue,
handleRailStyle,
}
},
render() {
const { t } = this
const { $t } = this
return (
<NSpace justify="center">
@ -79,8 +76,8 @@ const ThemeSwitch = defineComponent({
),
default: () =>
this.themeValue
? t('headerSettingOptions.ThemeOptions.Dark')
: t('headerSettingOptions.ThemeOptions.Light'),
? $t('headerSettingOptions.ThemeOptions.Dark')
: $t('headerSettingOptions.ThemeOptions.Light'),
}}
</NTooltip>
</NSpace>

View File

@ -15,7 +15,6 @@ import ThemeSwitch from '@/layout/components/SiderBar/components/SettingDrawer/c
import { APP_THEME } from '@/app-config/designConfig'
import { useSetting } from '@/store'
import { useI18n } from '@/hooks/web/index'
import type { PropType } from 'vue'
import type { Placement } from '@/types/modules/component'
@ -38,7 +37,6 @@ const SettingDrawer = defineComponent({
},
emits: ['update:show'],
setup(props, { emit }) {
const { t } = useI18n()
const settingStore = useSetting()
const { changePrimaryColor, changeSwitcher, updateContentTransition } =
@ -80,7 +78,6 @@ const SettingDrawer = defineComponent({
return {
modelShow,
t,
changePrimaryColor,
themeValue,
primaryColorOverride,
@ -95,7 +92,7 @@ const SettingDrawer = defineComponent({
}
},
render() {
const { t } = this
const { $t } = this
return (
<NDrawer
@ -103,14 +100,14 @@ const SettingDrawer = defineComponent({
placement={this.placement}
width={this.width}
>
<NDrawerContent title={t('headerSettingOptions.Title')}>
<NDrawerContent title={$t('headerSettingOptions.Title')}>
<NSpace class="setting-drawer__space" vertical>
<NDivider titlePlacement="center">
{t('headerSettingOptions.ThemeOptions.Title')}
{$t('headerSettingOptions.ThemeOptions.Title')}
</NDivider>
<ThemeSwitch />
<NDivider titlePlacement="center">
{t('headerSettingOptions.ThemeOptions.PrimaryColorConfig')}
{$t('headerSettingOptions.ThemeOptions.PrimaryColorConfig')}
</NDivider>
<NColorPicker
swatches={APP_THEME.appThemeColors}
@ -118,7 +115,7 @@ const SettingDrawer = defineComponent({
onUpdateValue={this.changePrimaryColor.bind(this)}
/>
<NDivider titlePlacement="center">
{t('headerSettingOptions.ContentTransition')}
{$t('headerSettingOptions.ContentTransition')}
</NDivider>
<NSelect
v-model:value={this.contentTransition}
@ -128,7 +125,7 @@ const SettingDrawer = defineComponent({
}}
/>
<NDivider titlePlacement="center">
{t('headerSettingOptions.InterfaceDisplay')}
{$t('headerSettingOptions.InterfaceDisplay')}
</NDivider>
<NDescriptions labelPlacement="left" column={1}>
<NDescriptionsItem label="多标签">

View File

@ -35,8 +35,10 @@ import {
createLeftIconOptions,
createRightIconOptions,
} from './hook'
import { useDevice, useFullscreen } from '@/hooks/web/index'
import { useDevice } from '@/hooks/web/index'
import { globalVariableToRefs, setVariable } from '@/hooks/variable/index'
import { useFullscreen } from 'vue-hooks-plus'
import { useI18n } from '@/hooks/web/index'
import type { IconEventMapOptions, IconEventMap } from './type'
@ -46,8 +48,9 @@ export default defineComponent({
const settingStore = useSetting()
const { updateLocale, changeSwitcher } = settingStore
const { t } = useI18n()
const [isFullscreen, { toggleFullscreen }] = useFullscreen(
const [isFullscreen, { toggleFullscreen, isEnabled }] = useFullscreen(
document.getElementsByTagName('html')[0],
)
const { drawerPlacement, breadcrumbSwitch, reloadRouteSwitch } =
@ -96,6 +99,10 @@ export default defineComponent({
window.open('https://github.com/XiaoDaiGua-Ray/ray-template')
},
fullscreen: () => {
if (!isEnabled) {
window.$message.warning(t('globalMessage.isEnabledFullscreen'))
}
toggleFullscreen()
},
search: () => {

View File

@ -0,0 +1,3 @@
{
"isEnabledFullscreen": "The current environment does not support full screen"
}

View File

@ -0,0 +1,3 @@
{
"isEnabledFullscreen": "当前环境不支持全屏"
}

View File

@ -49,7 +49,11 @@ export const downloadBase64File = (base64: string, fileName: string) => {
link.href = base64
link.download = fileName
link.style.display = 'none'
document.body.appendChild(link)
link.click()
document.body.removeChild(link)
}
/**
@ -135,12 +139,10 @@ export const downloadAnyFile = (
link.style.display = 'none'
document.body.appendChild(link)
link.click()
document.body.removeChild(link)
URL.revokeObjectURL(url)
resolve()
})
}

View File

@ -1,2 +1,3 @@
export { call } from './call'
export { unrefElement } from './unrefElement'
export { renderNode } from './renderNode'

View File

@ -0,0 +1,53 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-10-27
*
* @workspace ray-template
*
* @remark
*/
import { isValueType } from '@/utils/basic'
import type { VNode, VNodeChild } from 'vue'
export type RenderVNodeType =
| VNode
| VNodeChild
| (() => VNode)
| string
| number
| undefined
| null
| JSX.Element
export type DefaultElement = NonNullable<
Omit<RenderVNodeType, 'string' | 'number'>
>
export interface RenderNodeOptions<T extends DefaultElement> {
defaultElement?: T
}
export function renderNode<T extends DefaultElement>(
vnode: RenderVNodeType,
options?: RenderNodeOptions<T>,
) {
if (!vnode) {
const { defaultElement } = options ?? {}
return typeof defaultElement === 'function'
? defaultElement
: () => defaultElement
}
if (typeof vnode === 'string' || isValueType<object>(vnode, 'Object')) {
return () => vnode
}
if (typeof vnode === 'function') {
return vnode
}
}

View File

@ -1,12 +1,9 @@
import './index.scss'
import { NCard, NSwitch, NSpace, NP, NH2, NButton } from 'naive-ui'
import { NCard, NSwitch, NSpace, NH2, NButton } from 'naive-ui'
import RChart from '@/components/RChart/index'
import dayjs from 'dayjs'
import type { ECharts } from 'echarts/core'
import type { RayChartInst } from '@/components/RChart/index'
import type { RayChartInst } from '@/components/RChart/type'
const Echart = defineComponent({
name: 'REchart',
@ -85,9 +82,6 @@ const Echart = defineComponent({
],
}
const baseLineOptions = ref({
title: {
text: dayjs().valueOf(),
},
tooltip: {
trigger: 'axis',
axisPointer: {
@ -100,11 +94,6 @@ const Echart = defineComponent({
legend: {
data: ['Email', 'Union Ads', 'Video Ads', 'Direct', 'Search Engine'],
},
toolbox: {
feature: {
saveAsImage: {},
},
},
grid: {
left: '3%',
right: '4%',
@ -189,16 +178,6 @@ const Echart = defineComponent({
chartAria.value = bool
}
const handleChartRenderSuccess = (chart: ECharts) => {
window.$notification.info({
title: '可视化图渲染成功回调函数',
content: '可视化图渲染成功, 并且返回了当前可视化图实例',
duration: 5 * 1000,
})
console.log(baseChartRef.value, chart)
}
const mountChart = () => {
baseChartRef.value?.render()
}
@ -208,8 +187,6 @@ const Echart = defineComponent({
}
const handleUpdateTitle = () => {
baseLineOptions.value.title.text = dayjs().valueOf()
const createData = () => Math.floor((Math.random() + 1) * 100)
baseLineOptions.value.series[0].data = new Array(7)
@ -227,7 +204,6 @@ const Echart = defineComponent({
handleLoadingShow,
chartAria,
handleAriaShow,
handleChartRenderSuccess,
basePieOptions,
baseLineOptions,
...toRefs(state),
@ -262,7 +238,7 @@ const Echart = defineComponent({
</li>
</ul>
</NCard>
<NH2>animation</NH2>
<NH2>animation card </NH2>
<NSpace style={['padding: 18px 0']}>
<NButton onClick={this.mountChart.bind(this)}></NButton>
<NButton onClick={this.unmountChart.bind(this)}></NButton>
@ -272,10 +248,12 @@ const Echart = defineComponent({
</NSpace>
<div class="chart--container">
<RChart
title="周销售量"
ref="baseChartRef"
autoChangeTheme
options={this.baseLineOptions}
showAria={this.chartAria}
preset="card"
/>
</div>
<NH2></NH2>

View File

@ -1,15 +1,15 @@
import { NForm, NFormItem, NInput, NButton, NSpace, NDivider } from 'naive-ui'
import { NForm, NFormItem, NInput, NButton } from 'naive-ui'
import { setStorage } from '@/utils/cache'
import { useSignin } from '@/store'
import { useI18n } from '@/hooks/web/index'
import { APP_CATCH_KEY, ROOT_ROUTE } from '@/app-config/appConfig'
import { useVueRouter } from '@/hooks/web/index'
import { setVariable } from '@/hooks/variable/index'
import { setVariable, globalVariableToRefs } from '@/hooks/variable/index'
import type { FormInst } from 'naive-ui'
const Signin = defineComponent({
export default defineComponent({
name: 'RSignin',
setup() {
const loginFormRef = ref<FormInst>()
@ -19,6 +19,7 @@ const Signin = defineComponent({
const { signin } = signinStore
const { path } = ROOT_ROUTE
const globalSpinning = globalVariableToRefs('globalSpinning')
const useSigninForm = () => ({
name: 'Ray Admin',
@ -74,37 +75,37 @@ const Signin = defineComponent({
loginFormRef,
handleLogin,
rules,
t,
globalSpinning,
}
},
render() {
const { t } = this
const { $t, globalSpinning } = this
return (
<NForm model={this.signinForm} ref="loginFormRef" rules={this.rules}>
<NFormItem label={t('views.login.index.Name')} path="name">
<NFormItem label={$t('views.login.index.Name')} path="name">
<NInput
v-model:value={this.signinForm.name}
placeholder={t('views.login.index.NamePlaceholder')}
placeholder={$t('views.login.index.NamePlaceholder')}
/>
</NFormItem>
<NFormItem label={t('views.login.index.Password')} path="pwd">
<NFormItem label={$t('views.login.index.Password')} path="pwd">
<NInput
v-model:value={this.signinForm.pwd}
type="password"
placeholder={t('views.login.index.PasswordPlaceholder')}
showPasswordOn="click"
placeholder={$t('views.login.index.PasswordPlaceholder')}
/>
</NFormItem>
<NButton
style={['width: 100%', 'margin-to: 18px']}
type="primary"
onClick={this.handleLogin.bind(this)}
loading={globalSpinning}
>
{t('views.login.index.Login')}
{$t('views.login.index.Login')}
</NButton>
</NForm>
)
},
})
export default Signin

View File

@ -21,13 +21,11 @@ import ThemeSwitch from '@/layout/components/SiderBar/components/SettingDrawer/c
import { useSetting } from '@/store'
import { LOCAL_OPTIONS } from '@/app-config/localConfig'
import { useI18n } from '@/hooks/web/index'
import { useWindowSize } from '@vueuse/core'
const Login = defineComponent({
name: 'RLogin',
setup() {
const { t } = useI18n()
const {
layout: { copyright },
} = __APP_CFG__
@ -44,13 +42,12 @@ const Login = defineComponent({
...toRefs(state),
windowHeight,
updateLocale,
t,
copyright,
windowWidth,
}
},
render() {
const { t } = this
const { $t } = this
return (
<div
@ -125,19 +122,19 @@ const Login = defineComponent({
default: () => (
<>
<NTabPane
tab={t('views.login.index.Signin')}
tab={$t('views.login.index.Signin')}
name="signin"
>
<Signin />
</NTabPane>
<NTabPane
tab={t('views.login.index.Register')}
tab={$t('views.login.index.Register')}
name="register"
>
<Register />
</NTabPane>
<NTabPane
tab={t('views.login.index.QRCodeSignin')}
tab={$t('views.login.index.QRCodeSignin')}
name="qrcodeSignin"
>
<QRCodeSignin />

View File

@ -70,7 +70,6 @@
"useCssModule": true,
"useCssVars": true,
"useDialog": true,
"useI18n": true,
"useLink": true,
"useLoadingBar": true,
"useMessage": true,

View File

@ -66,7 +66,6 @@ declare global {
const useCssModule: typeof import('vue')['useCssModule']
const useCssVars: typeof import('vue')['useCssVars']
const useDialog: typeof import('naive-ui')['useDialog']
const useI18n: typeof import('vue-i18n')['useI18n']
const useLink: typeof import('vue-router')['useLink']
const useLoadingBar: typeof import('naive-ui')['useLoadingBar']
const useMessage: typeof import('naive-ui')['useMessage']

View File

@ -63,7 +63,6 @@ export default function (mode: string): PluginOption[] {
'vue',
'vue-router',
'pinia',
'vue-i18n',
{
'naive-ui': [
'useDialog',
@ -110,11 +109,6 @@ export default function (mode: string): PluginOption[] {
libDirectory: '',
camel2DashComponentName: false,
},
{
libName: 'lodash',
libDirectory: '',
camel2DashComponentName: false,
},
],
}),
{