diff --git a/src/locales/lang/en-US/menu.json b/src/locales/lang/en-US/menu.json
index 2b1bd1b7..77ece1c8 100644
--- a/src/locales/lang/en-US/menu.json
+++ b/src/locales/lang/en-US/menu.json
@@ -24,5 +24,6 @@
"TemplateHooks": "Template Api",
"Modal": "Modal",
"ContextMenu": "Right Click Menu",
- "CacheDemo": "Cache Utils Demo"
+ "CacheDemo": "Cache Utils Demo",
+ "Form": "Form"
}
diff --git a/src/locales/lang/zh-CN/menu.json b/src/locales/lang/zh-CN/menu.json
index e6624546..fda97e14 100644
--- a/src/locales/lang/zh-CN/menu.json
+++ b/src/locales/lang/zh-CN/menu.json
@@ -24,5 +24,6 @@
"TemplateHooks": "模板内置 Api",
"Modal": "模态框",
"ContextMenu": "右键菜单",
- "CacheDemo": "缓存工具函数"
+ "CacheDemo": "缓存工具函数",
+ "Form": "表单"
}
diff --git a/src/router/modules/demo/cache-demo.ts b/src/router/modules/demo/cache-demo.ts
index f3a5f3e3..f9dfdbee 100644
--- a/src/router/modules/demo/cache-demo.ts
+++ b/src/router/modules/demo/cache-demo.ts
@@ -11,9 +11,6 @@ const cacheDemo: AppRouteRecordRaw = {
i18nKey: t('menu.CacheDemo'),
icon: 'other',
order: 3,
- extra: {
- label: 'new',
- },
},
}
diff --git a/src/router/modules/demo/echart.ts b/src/router/modules/demo/echart.ts
index 3706de57..5ec03d9a 100644
--- a/src/router/modules/demo/echart.ts
+++ b/src/router/modules/demo/echart.ts
@@ -11,6 +11,9 @@ const echart: AppRouteRecordRaw = {
i18nKey: t('menu.Echart'),
icon: 'echart',
order: 1,
+ extra: {
+ label: 'useChart',
+ },
},
}
diff --git a/src/router/modules/demo/form.ts b/src/router/modules/demo/form.ts
new file mode 100644
index 00000000..53467a63
--- /dev/null
+++ b/src/router/modules/demo/form.ts
@@ -0,0 +1,20 @@
+import { t } from '@/hooks/web/useI18n'
+import { LAYOUT } from '@/router/constant'
+
+import type { AppRouteRecordRaw } from '@/router/types'
+
+const form: AppRouteRecordRaw = {
+ path: '/form',
+ name: 'FormView',
+ component: () => import('@/views/demo/form'),
+ meta: {
+ i18nKey: t('menu.Form'),
+ icon: 'other',
+ order: 2,
+ extra: {
+ label: 'useForm',
+ },
+ },
+}
+
+export default form
diff --git a/src/router/modules/demo/mock.ts b/src/router/modules/demo/mock.ts
index 1935a6ac..0b17c934 100644
--- a/src/router/modules/demo/mock.ts
+++ b/src/router/modules/demo/mock.ts
@@ -12,6 +12,9 @@ const mockDemo: AppRouteRecordRaw = {
icon: 'other',
order: 3,
keepAlive: false,
+ extra: {
+ label: 'usePagination',
+ },
},
}
diff --git a/src/router/modules/demo/table.ts b/src/router/modules/demo/table.ts
index e7e1a045..c9a2148d 100644
--- a/src/router/modules/demo/table.ts
+++ b/src/router/modules/demo/table.ts
@@ -11,6 +11,9 @@ const table: AppRouteRecordRaw = {
i18nKey: t('menu.Table'),
icon: 'other',
order: 2,
+ extra: {
+ label: 'useTable',
+ },
},
}
diff --git a/src/store/modules/menu/index.ts b/src/store/modules/menu/index.ts
index cf452454..fde9049e 100644
--- a/src/store/modules/menu/index.ts
+++ b/src/store/modules/menu/index.ts
@@ -136,9 +136,8 @@ export const piniaMenuStore = defineStore(
default: () => label.value,
}),
breadcrumbLabel: label.value,
- /** 检查该菜单项是否展示 */
} as AppMenuOption
- /** 合并 icon */
+ /** 合并 icon, extra */
const attr: AppMenuOption = Object.assign({}, route, {
icon: createMenuIcon(option),
extra: createMenuExtra(option),
diff --git a/src/types/modules/utils.ts b/src/types/modules/utils.ts
index 3ab0b032..01579fc6 100644
--- a/src/types/modules/utils.ts
+++ b/src/types/modules/utils.ts
@@ -92,6 +92,8 @@ export type CipherParams = CryptoJS.lib.CipherParams
export type AnyFC = (...args: P[]) => R
+export type VoidFC = (...args: any[]) => void
+
export type PartialCSSStyleDeclaration = Partial<
Record
>
diff --git a/src/views/demo/echart/index.tsx b/src/views/demo/echart/index.tsx
index 29a49b50..196818f6 100644
--- a/src/views/demo/echart/index.tsx
+++ b/src/views/demo/echart/index.tsx
@@ -3,12 +3,25 @@ import './index.scss'
import { NCard, NSwitch, NFlex, NH2, NButton } from 'naive-ui'
import { RChart } from '@/components'
+import { useChart } from '@/components'
+
import type { RChartType } from '@/components'
const Echart = defineComponent({
name: 'REchart',
setup() {
- const baseChartRef = ref()
+ const [register, { getChartInstance, dispose, render, isDispose }] =
+ useChart()
+ const [
+ register2,
+ {
+ getChartInstance: getChartInstance2,
+ dispose: dispose2,
+ render: render2,
+ isDispose: isDispose2,
+ },
+ ] = useChart()
+
const chartLoading = ref(false)
const chartAria = ref(false)
const state = reactive({
@@ -179,15 +192,15 @@ const Echart = defineComponent({
}
const mountChart = () => {
- if (!baseChartRef.value?.isDispose()) {
- baseChartRef.value?.render()
+ if (isDispose()) {
+ render()
} else {
window.$message.warning('图表已经渲染')
}
}
const unmountChart = () => {
- baseChartRef.value?.dispose()
+ dispose()
}
const updateChartOptions = () => {
@@ -203,7 +216,6 @@ const Echart = defineComponent({
return {
baseOptions,
- baseChartRef,
chartLoading,
handleLoadingShow,
chartAria,
@@ -214,9 +226,16 @@ const Echart = defineComponent({
mountChart,
unmountChart,
updateChartOptions,
+ register,
+ register2,
+ dispose2,
+ render2,
+ isDispose2,
}
},
render() {
+ const { register, register2, dispose2, render2, isDispose2 } = this
+
return (
@@ -246,6 +265,9 @@ const Echart = defineComponent({
属性,只有元素在可见范围才会渲染图表,可以滚动查看效果
+
+ 7. useChart 方法
+
@@ -258,8 +280,8 @@ const Echart = defineComponent({
-
-
-
+
+
+ {
+ if (isDispose2()) {
+ render2()
+ } else {
+ window.$message.warning('图表已经渲染')
+ }
+ }}
+ >
+ 渲染
+
+ 卸载
+
+
+
+
+
+ *
+ * @date 2024-03-27
+ *
+ * @workspace ray-template
+ *
+ * @remark 今天也是元气满满撸代码的一天
+ */
+
+import { RForm } from '@/components'
+import {
+ NFormItemGi,
+ NDatePicker,
+ NGrid,
+ NInput,
+ NInputNumber,
+ NFlex,
+ NButton,
+ NRadio,
+ NRadioGroup,
+} from 'naive-ui'
+
+import { useForm } from '@/components'
+
+import type { RFormRules } from '@/components'
+
+export default defineComponent({
+ name: 'RFormDemo',
+ setup() {
+ // 使用以下 hooks 的时候,应该注意调用时机
+ const [
+ register,
+ { getFormInstance, validate, restoreValidation, formModel, formRules },
+ ] = useForm(
+ {
+ name: null,
+ age: null,
+ gender: null,
+ date: null,
+ remark: null,
+ },
+ {
+ name: {
+ required: true,
+ message: '请输入姓名',
+ trigger: ['blur', 'change'],
+ },
+ date: {
+ required: true,
+ message: '请选择日期',
+ trigger: ['blur', 'change'],
+ type: 'number',
+ },
+ gender: {
+ required: true,
+ message: '请选择性别',
+ trigger: 'change',
+ },
+ age: {
+ required: true,
+ message: '请输入年龄',
+ trigger: ['blur', 'change'],
+ type: 'number',
+ },
+ },
+ )
+
+ /**
+ *
+ * @description
+ * 如果待验证数据类型为: number, array 等,需要手动设置 type 类型。
+ * 具体可以吃查看: async-validator type
+ * @see https://github.com/yiminghe/async-validator?tab=readme-ov-file#type
+ *
+ * 如果你需要自定义验证,可以查看:naive ui custom validation
+ * @see https://www.naiveui.com/zh-CN/dark/components/form#custom-validation.vue
+ *
+ * 如果只是简单的 rules 管理,可以在初始化 useForm 的时候传入第二个参数;
+ * 然后使用 formRules 方法获取到初始化 rules 数据。
+ */
+ const rules = ref(formRules())
+ /**
+ *
+ * @description
+ * 如果只是简单的数据,可以在初始化 useForm 的时候直接传入第一个参数;
+ * 然后使用 formModel 方法获取到初始化 model 数据。
+ *
+ * 动态的复杂数据,不建议使用该方法管理 model;手动的拆分出来是一个更加好的选择。
+ */
+ const condition = ref(formModel())
+
+ return {
+ register,
+ rules,
+ condition,
+ restoreValidation,
+ formModel,
+ validate,
+ }
+ },
+ render() {
+ const { rules } = this
+ const { register, restoreValidation, formModel, validate } = this
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 女
+ 男
+
+
+
+
+
+
+
+ {
+ this.condition = formModel()
+ }}
+ >
+ 重置表单为初始状态
+
+
+ 移除校验状态
+
+ validate()}>
+ 校验
+
+
+
+
+
+ )
+ },
+})
diff --git a/src/views/demo/mock-demo/index.tsx b/src/views/demo/mock-demo/index.tsx
index 630bab62..ef53550c 100644
--- a/src/views/demo/mock-demo/index.tsx
+++ b/src/views/demo/mock-demo/index.tsx
@@ -14,6 +14,7 @@ import { RCollapseGrid, RTable } from '@/components'
import { useHookPlusRequest } from '@/axios'
import { getPersonList } from '@/api/demo/mock/person'
+import { usePagination } from '@/hooks'
import type { Person } from '@/api/demo/mock/person'
@@ -86,24 +87,23 @@ const MockDemo = defineComponent({
const condition = reactive({
email: null,
})
- const paginationRef = reactive({
- page: 1,
- pageSize: 10,
- itemCount: 0,
- pageSizes: [10, 20, 30, 40, 50],
- showSizePicker: true,
- onUpdatePage: (page: number) => {
- paginationRef.page = page
- getPerson()
- },
- onUpdatePageSize: (pageSize: number) => {
- paginationRef.pageSize = pageSize
- paginationRef.page = 1
-
- getPerson()
- },
+ const {
+ getPagination,
+ getPage,
+ getPageSize,
+ setItemCount,
+ getCallback,
+ setPage,
+ setPageSize,
+ } = usePagination(() => {
+ personFetchRun({
+ page: getPage(),
+ pageSize: getPageSize(),
+ email: condition.email,
+ })
})
+ const paginationRef = getPagination()
const {
data: personData,
loading: personLoading,
@@ -111,36 +111,26 @@ const MockDemo = defineComponent({
} = useHookPlusRequest(getPersonList, {
defaultParams: [
{
- page: paginationRef.page,
- pageSize: paginationRef.pageSize,
+ page: getPage(),
+ pageSize: getPageSize(),
email: condition.email,
},
],
onSuccess: (res) => {
const { total } = res
- paginationRef.itemCount = total
+ setItemCount(total)
},
})
- const getPerson = () => {
- const { pageSize, page } = paginationRef
- const { email } = condition
-
- personFetchRun({
- page,
- pageSize,
- email,
- })
- }
-
return {
personData,
personLoading,
- paginationRef,
+ getPagination,
columns,
...toRefs(condition),
- getPerson,
+ getCallback,
+ paginationRef,
}
},
render() {
@@ -170,7 +160,7 @@ const MockDemo = defineComponent({
>
),
action: () => (
-
+
搜索
),
diff --git a/src/views/demo/modal-demo/index.tsx b/src/views/demo/modal-demo/index.tsx
index a5476adc..64fe1511 100644
--- a/src/views/demo/modal-demo/index.tsx
+++ b/src/views/demo/modal-demo/index.tsx
@@ -12,6 +12,8 @@
import { RModal } from '@/components'
import { NButton, NCard, NFlex } from 'naive-ui'
+import { useModal } from '@/components'
+
export default defineComponent({
name: 'ModalDemo',
setup() {
@@ -20,12 +22,36 @@ export default defineComponent({
modal2: false,
modal3: false,
})
+ const { create } = useModal()
+
+ const createCardModal = () => {
+ create({
+ title: '卡片模态框',
+ dad: true,
+ preset: 'card',
+ content: '我可以被拖拽的全屏card模态框',
+ fullscreen: true,
+ })
+ }
+
+ const createDialogModal = () => {
+ create({
+ title: '模态框',
+ content: '内容',
+ preset: 'dialog',
+ dad: true,
+ })
+ }
return {
...toRefs(state),
+ createCardModal,
+ createDialogModal,
}
},
render() {
+ const { createCardModal, createDialogModal } = this
+
return (
@@ -91,6 +117,12 @@ export default defineComponent({
所有的宽度配置属性都会注入一个对应的 `css variable`,有时候会用上。
+
+ 创建卡片模态框
+
+ 创建对话框模态框
+
+
)
},
diff --git a/src/views/demo/table/index.tsx b/src/views/demo/table/index.tsx
index e3d86d49..36b45a9d 100644
--- a/src/views/demo/table/index.tsx
+++ b/src/views/demo/table/index.tsx
@@ -23,20 +23,29 @@ import {
} from 'naive-ui'
import { RCollapseGrid, RTable, RIcon, RMoreDropdown } from '@/components'
+import { uuid } from '@/utils'
+import { useTable } from '@/components'
+
import type { DataTableColumns } from 'naive-ui'
-import type { RTableType } from '@/components'
type RowData = {
- key: number
+ key: number | string
name: string
age: number
address: string
tags: string[]
+ remark: string
}
const TableView = defineComponent({
name: 'TableView',
setup() {
+ // 使用以下 hooks 的时候,应该注意调用时机
+ const [
+ register,
+ { getTableInstance, clearFilters, clearSorter, scrollTo, filters, sort },
+ ] = useTable()
+
const baseColumns = [
{
title: 'Name',
@@ -68,7 +77,7 @@ const TableView = defineComponent({
{
title: 'Remark',
key: 'remark',
- width: 300,
+ width: 100,
},
{
title: 'Action',
@@ -98,32 +107,7 @@ const TableView = defineComponent({
const actionColumns = ref>(
[...baseColumns].map((curr) => ({ ...curr, width: 400 })),
)
- const tableData = ref([
- {
- key: 0,
- name: 'John Brown',
- age: 32,
- address: 'New York No. 1 Lake Park',
- tags: ['nice', 'developer'],
- remark: '我是一条很长很长的备注',
- },
- {
- key: 1,
- name: 'Jim Green',
- age: 42,
- address: 'London No. 1 Lake Park',
- tags: ['wow'],
- remark: '我是一条很长很长的备注',
- },
- {
- key: 2,
- name: 'Joe Black',
- age: 32,
- address: 'Sidney No. 1 Lake Park',
- tags: ['cool', 'teacher'],
- remark: '我是一条很长很长的备注',
- },
- ])
+ const tableData = ref([])
const tableMenuOptions = [
{
label: '编辑',
@@ -140,20 +124,38 @@ const TableView = defineComponent({
tableLoading: false,
})
- const handleMenuSelect = (key: string | number) => {
+ const createTableData = () => {
+ for (let i = 0; i < 20; i++) {
+ tableData.value.push({
+ key: uuid(),
+ name: 'John Brown',
+ age: i + 20,
+ address: 'New York No. 1 Lake Park',
+ tags: ['nice', 'developer'],
+ remark: '我是一条很长很长的备注',
+ })
+ }
+ }
+
+ const menuSelect = (key: string | number) => {
window.$message.info(`${key}`)
}
+ createTableData()
+
return {
...toRefs(state),
tableData,
actionColumns,
baseColumns,
tableMenuOptions,
- handleMenuSelect,
+ menuSelect,
+ register,
}
},
render() {
+ const { register } = this
+
return (
@@ -199,7 +201,8 @@ const TableView = defineComponent({
}}
标题插槽:
@@ -215,7 +218,7 @@ const TableView = defineComponent({
}}
contextMenuOptions={this.tableMenuOptions}
loading={this.tableLoading}
- onContextMenuClick={this.handleMenuSelect.bind(this)}
+ onContextMenuClick={this.menuSelect.bind(this)}
toolOptions={[
{{
diff --git a/vitest.config.ts b/vitest.config.ts
index e7713ec1..04a310df 100644
--- a/vitest.config.ts
+++ b/vitest.config.ts
@@ -18,6 +18,14 @@ export default defineConfig((configEnv) =>
environment: 'happy-dom',
globals: true,
poolOptions: {
+ /**
+ *
+ * 如此配置是为避免: Module did not self-register... 错误;
+ * 该错误是一个历史悠久遗留问题,可以查看该 issues:
+ * @see https://github.com/vitest-dev/vitest/issues/740
+ *
+ * 目前暂时没有更好的解决方案,这么做会导致单测执行速度变慢,但是可以避免错误,后续有更好的解决方案会更新。
+ */
threads: {
maxThreads: 1,
minThreads: 0,