活动介绍

vue3+ts中的hook中通过ref定义了form,在页面中使用hook,怎么在方法中使用form

时间: 2025-10-21 15:46:12 AIGC 浏览: 2
在 Vue 3 + TypeScript 的 hook 里用 `ref` 定义 `form`,在页面使用此 hook 时,可按以下步骤在方法中使用 `form`。 ### 1. 创建 hook 首先创建一个 hook 文件,在其中用 `ref` 定义 `form`。 ```typescript // useMyForm.ts import { ref } from 'vue'; export function useMyForm() { const form = ref<HTMLFormElement | null>(null); const submitForm = () => { if (form.value) { form.value.submit(); } }; return { form, submitForm }; } ``` ### 2. 在页面组件中使用 hook 在页面组件里引入并使用这个 hook,还能在方法中操作 `form`。 ```vue <template> <form ref="form"> <input type="text" name="username" /> <button @click="submitForm">提交表单</button> </form> </template> <script lang="ts" setup> import { useMyForm } from './useMyForm'; const { form, submitForm } = useMyForm(); // 其他方法中使用 form const handleOtherAction = () => { if (form.value) { // 可以在这里对 form 进行其他操作 console.log(form.value.elements); } }; </script> ``` ### 代码解释 - 在 `useMyForm` hook 中,用 `ref` 定义了 `form`,其初始值为 `null`。 - `submitForm` 方法会检查 `form.value` 是否存在,若存在就调用 `submit` 方法提交表单。 - 在页面组件中引入 `useMyForm` hook,解构出 `form` 和 `submitForm`。 - 在 `handleOtherAction` 方法里,同样检查 `form.value` 是否存在,若存在就可对 `form` 进行其他操作。
阅读全文

相关推荐

import { computed, ref } from "vue"; import { getEquipNameToDeptMap, getDeptByEquipName, clearequipCache } from "@/service/api/"; // 根据实际路径调整 /** * 设备部门映射钩子 * 提供设备名称到部门的映射关系操作 */ export function useEquipequip() { // 加载状态 const loading = ref(false); // 错误信息 const error = ref<Error | null>(null); // 设备名称到部门的映射 const nameToequipMap = ref<Record<string, string>>({}); // 设备选项列表(新增) const equipOptions = ref<{ label: string; value: string }[]>([]); // 搜索查询(新增) const equipSearchQuery = ref(""); // 计算过滤后的设备选项(新增) const filteredequipOptions = computed(() => { if (!equipSearchQuery.value) return equipOptions.value; const query = equipSearchQuery.value.toLowerCase(); return equipOptions.value.filter(option => option.label.toLowerCase().includes(query) ); }); // 处理设备搜索(新增) const handleequipSearch = (query: string) => { equipSearchQuery.value = query; }; /** * 加载设备映射关系 * @param forceRefresh 是否强制刷新缓存 */ const loadEquipMap = async (forceRefresh: boolean = false) => { try { loading.value = true; error.value = null; nameToequipMap.value = await getEquipNameToDeptMap(forceRefresh); // 生成设备选项列表(新增) equipOptions.value = Object.keys(nameToequipMap.value).map(name => ({ label: name, value: name })); } catch (err) { console.error('加载设备映射失败:', err); error.value = err as Error; } finally { loading.value = false; } }; /** * 根据设备名称获取部门 * @param equipName 设备名称 * @returns 对应的部门值 */ const getequip = async (equipName: string): Promise<string | undefined> => { // 如果本地已有映射,直接使用 if (nameToequipMap.value[equipName]) { return nameToequipMap.value[equipName]; } // 否则从API获取 try { loading.value = true; return await getDeptByEquipName(equipName); } catch (err) { console.error('获取设备部门失败:', err); error.value = err as Error; return undefined; } finally { loading.value = false; } }; /** * 刷新设备映射缓存 */ const refreshCache = () => { clearequipCache(); loadEquipMap(true); }; // 返回钩子方法和状态 return { loading, error, nameToequipMap, equipOptions, // 新增 filteredequipOptions, // 新增 equipSearchQuery, // 新增 loadEquipMap, getequip, refreshCache, handleequipSearch // 新增 }; } 根据这个hook修改下面代码,实现编辑和克隆时,带出已有信息,并在设备选择与部门自动填充功能 <script setup lang="ts"> import { computed, reactive, ref, watch } from "vue"; import { useFormRules, useNaiveForm } from "@/hooks/common/form"; import { fetchAddInterlock, fetchUpdateInterlock, } from "@/service/api"; import { $t } from "@/locales"; import { lockTypesOptions, logLevelTypeOptions, moduleTypesOptions, } from "@/constants/business"; import { translateOptions } from "@/utils/common"; import { NSelect } from "naive-ui"; import { useEquipequip } from "@/hooks/common/useEquipment"; defineOptions({ name: "LedgerOperateDrawer", }); interface Props { /** the type of operation */ operateType: NaiveUI.TableOperateType; /** the edit row data */ rowData?: Api.InterLock.Ledger | null; } const props = defineProps(); interface Emits { (e: "submitted"): void; } const emit = defineEmits<Emits>(); const visible = defineModel<boolean>("visible", { default: false, }); // 设备搜索查询 const equipSearchQuery = ref(""); const { formRef, validate, restoreValidation } = useNaiveForm(); const { defaultRequiredRule } = useFormRules(); const title = computed(() => { const titles: Record<NaiveUI.TableOperateType, string> = { add: $t("common.add"), edit: $t("common.edit"), clone: $t("common.clone"), }; return titles[props.operateType]; }); const model: Api.InterLock.LedgerUpdateParams = reactive(createDefaultModel()); function createDefaultModel(): Api.InterLock.LedgerAddParams { return { alarmName: "", lockType: "sinterlock", moduleName: "", moduleType: "", product: "", deptName: "", trigger: "", channelNumber: "", triggerWay: "", alarmDesc: "", alarmLevel: "INFO", alarmDetail: "", alarmAction: "", alarmChannel: "", alarmActionDesc: "", condition: "", desc: "", }; } type RuleKey = Extract< keyof Api.InterLock.LedgerUpdateParams, | "alarmName" | "lockType" | "moduleName" | "moduleType" | "product" >; const rules = ref<Record<RuleKey, App.Global.FormRule>>({ alarmName: defaultRequiredRule, lockType: defaultRequiredRule, moduleName: defaultRequiredRule, moduleType: defaultRequiredRule, product: defaultRequiredRule, }); // 设备选择变化处理 const handleProductChange = async (productName: string) => { if (productName) { // 获取设备对应的部门 const dept = await getDept(productName); model.deptName = dept || ""; } else { model.deptName = ""; } }; function handleInitModel() { Object.assign(model, createDefaultModel()); const operateTypes = ["edit", "clone"]; if (operateTypes.includes(props.operateType) && props.rowData) { Object.assign(model, props.rowData); // 编辑模式:如果已有设备名称,自动填充部门 if (props.rowData.product) { // 使用异步方式获取部门(确保设备列表已加载) setTimeout(async () => { const dept = await getDept(props.rowData!.product); model.deptName = dept || ""; }, 100); } } } function closeDrawer() { visible.value = false; } async function handleSubmit() { await validate(); // request // 提交前确保部门信息已设置 if (model.product && !model.deptName) { const dept = await getDept(model.product); model.deptName = dept || ""; } if (props.operateType === "add" || props.operateType === "clone") { const { error } = await fetchAddInterlock(model); if (!error) { window.$message?.success($t("common.addSuccess")); } } else if (props.operateType === "edit") { const { error } = await fetchUpdateInterlock(model); if (!error) { window.$message?.success($t("common.updateSuccess")); } } closeDrawer(); emit("submitted"); } watch(visible, () => { if (visible.value) { handleInitModel(); restoreValidation(); } }); </script> <template> <NDrawer v-model:show="visible" display-directive="show" :width="360"> <NDrawerContent :title="title" :native-scrollbar="false" closable> <NForm ref="formRef" :model="model" :rules="rules"> <NFormItem :label="$t('page.interlock.ledger.alarmName')" path="alarmName" > <NInput v-model:value="model.alarmName" :placeholder="$t('page.interlock.ledger.form.alarmName')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.lockType')" path="lockType" > <NSelect v-model:value="model.lockType" :options="translateOptions(lockTypesOptions)" size="small" class="w-120px" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.product')" path="product"> <NSelect v-model:value="model.product" filterable clearable :placeholder="$t('page.interlock.ledger.form.product')" :options="filteredequipOptions" :loading="Loading" :remote="true" @update:value="handleProductChange" @search="handleequipSearch" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.moduleName')" path="moduleName" > <NInput v-model:value="model.moduleName" :placeholder="$t('page.interlock.ledger.form.moduleName')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.moduleType')" path="moduleType" > <NSelect v-model:value="model.moduleType" :options="translateOptions(moduleTypesOptions)" size="small" class="w-120px" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.deptName')" path="deptName" > <NInput v-model:value="model.deptName" :placeholder="$t('page.interlock.ledger.form.deptName')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.trigger')" path="trigger"> <NInput v-model:value="model.trigger" :placeholder="$t('page.interlock.ledger.form.trigger')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.triggerWay')" path="triggerWay" > <NInput v-model:value="model.triggerWay" :placeholder="$t('page.interlock.ledger.form.triggerWay')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.alarmDesc')" path="alarmDesc" > <NInput v-model:value="model.alarmDesc" :placeholder="$t('page.interlock.ledger.form.alarmDesc')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.alarmLevel')" path="alarmLevel" > <NSelect v-model:value="model.alarmLevel" :options="translateOptions(logLevelTypeOptions)" size="small" class="w-120px" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.alarmAction')" path="alarmAction" > <NInput v-model:value="model.alarmAction" :placeholder="$t('page.interlock.ledger.form.alarmAction')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.alarmChannel')" path="alarmChannel" > <NInput v-model:value="model.alarmChannel" :placeholder="$t('page.interlock.ledger.form.alarmChannel')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.channelNumber')" path="channelNumber" > <NInput v-model:value="model.channelNumber" :placeholder="$t('page.interlock.ledger.form.channelNumber')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.alarmActionDesc')"> <NInput v-model:value="model.alarmActionDesc" :placeholder="$t('page.interlock.ledger.form.alarmActionDesc')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.condition')"> <NInput v-model:value="model.condition" :placeholder="$t('page.interlock.ledger.form.condition')" /> </NFormItem> <NFormItem :label="$t('page.interlock.ledger.desc')"> <NInput v-model:value="model.desc" :placeholder="$t('page.interlock.ledger.form.desc')" /> </NFormItem> </NForm> <template #footer> <NSpace :size="16"> <NButton @click="closeDrawer">{{ $t("common.cancel") }}</NButton> <NButton type="primary" @click="handleSubmit">{{ $t("common.confirm") }}</NButton> </NSpace> </template> </NDrawerContent> </NDrawer> </template> <style scoped></style>

<template> <el-button type="primary" class="add-btn" @click="openDrawer('add', '新增非标准资产', [])">新增非标准资产</el-button> <el-input v-model="filterText" class="filterTextInput" placeholder="请输入" clearable /> <template v-if="filteredTreeData.length"> {{item.name}} 编辑 删除 </template> <el-button type="primary" class="add-btn" @click="openDrawer('childAdd', '新增字段', [])">新增字段</el-button> <template #operation="scope"> <el-button type="primary" link @click="openDrawer('childEdit','编辑字段', scope.row)">编辑</el-button> <el-button v-if="scope.row.status == 'enabled'" type="primary" style="color:#FF0000" link @click="changeStatus(scope.row)">禁用</el-button> <el-button v-if="scope.row.status == 'disabled'" type="primary" style="color:#49C625" link @click="changeStatus(scope.row)">启用</el-button> <el-button type="primary" link @click="deleteAccount(scope.row)">删除</el-button> </template> <el-empty description="请先选择非标准资产"></el-empty> <addNewFields ref="dialogAddNewFields" /> </template> <script setup lang="ts" name="standardProductFieldManage"> import { ref, reactive,onMounted ,watch} from "vue"; import { ElTree, FormInstance, FormRules, ElMessage } from "element-plus"; import { Supplier } from "@/api/interface"; import { ProTableInstance, ColumnProps } from "@/components/ProTable/interface"; import ProTable from "@/components/ProTable/index.vue"; import addNewFields from "@/views/system/standardProductFieldManage/addNewFields.vue"; import { nonstandardList, editNonstandard, editNonstandardClild, deleteNonstandard ,nonstandardClildList} from '@/api/modules/public'; import { da } from "element-plus/es/locale"; import { useHandleData } from "@/hooks/useHandleData"; // 初始化参数 const pageData = reactive({}) const treeData = ref<any[]>([]); const filteredTreeData = ref<any[]>([]); const activeTypeItem = ref<Record<string, any>>({}); const treeRef = ref<InstanceType<typeof ElTree>>(); const filterText = ref(""); // ProTable 实例 const proTable = ref(); const initParam = reactive({assetNonStandardId:''}); watch( () => filterText.value, (val) => { if (!val) { filteredTreeData.value = treeData.value; } else { const keyword = val.toLowerCase(); filteredTreeData.value = treeData.value.filter(item => item.name?.toLowerCase().includes(keyword) ); } }, { immediate: true } ); onMounted(async () => { getTreeList() }) // 左侧树、 const getTreeList = async () => { const data = await nonstandardList({}); treeData.value = data || []; filteredTreeData.value = data || [] } // 点击左侧,更新右侧列表 const handleItemClick = async (data: any) => { activeTypeItem.value = data; // proTable.value!.pageable.pageNum = 1; initParam.assetNonStandardId = data.id; } // 调用列表接口 const getTableList = async () => { const formData = { assetNonStandardId: activeTypeItem.value.id }; const response = await nonstandardClildList(formData); return { data: Array.isArray(response.data) ? response.data : { nonstandardList:[response.data] }, }; } // 返回列表数据,回显到table上 const dataCallback = (data: any) => { console.log('Array.isArray(data)',data) const deep = data.nonstandardList[0]; const lll = [ { code:'111111', id:'23' } ] return { data: lll, total: deep.totalCount || 0, pageNum: deep.pageCount || 1, pageSize: deep.pageSize || 10 } // return Array.isArray(data) ? data : [data]; // return { // dataListlist: data[0].dataList, // total: data[0].totalCount, // pageNum: data[0].pageNum, // pageSize: data[0].pageSize // } } // const dataCallback = (data: any) => { // console.log('data',data) // return data // } // 表格配置项 const columns = reactive<ColumnProps<Supplier.pageList>[]>([ { prop: "fieldName", label: "单位名称", }, ]) // 调用列表接口 // const getTableList = async () => { // console.log('activeTypeItem.value',activeTypeItem.value) // const formData = { // assetNonStandardId: activeTypeItem.value.id // }; // const response = await nonstandardClildList(formData); // console.log('response',response) // } // const getTableList = (params: any) => { // return nonstandardClildList(params); // } // // 返回列表数据,回显到table上 // const dataCallback = (data: any) => { // console.log('data',data) // return data // } // const refreshTable = () => { // // proTable.value!.pageable.pageNum = 1; // // proTable.value?.refreshData(); // }; // 新增,编辑 const dialogAddNewFields = ref<InstanceType<typeof addNewFields> | null>(null); const openDrawer = async (type: string, title: string, row: any) => { console.log('activeTypeItem.value',activeTypeItem.value) const params = { type: type, title: title, isView: false, row: {...row}, activeTypeItem:{...activeTypeItem.value}, api: type == 'add' || type == 'edit' ? editNonstandard : editNonstandardClild, getTableListData: getTreeList } dialogAddNewFields.value?.acceptParams(params); } // 删除资产 const handleDelType = async (params) => { await useHandleData(deleteNonstandard, { id: params.id }, 确认删除); getTreeList() } // 修改状态 const changeStatus = async (row: Supplier.pageList) => { } // 删除 const deleteAccount = async (row: Supplier.pageList) => { } </script> <style scoped lang="scss"> @use "@/styles/treeIndex.scss"; .list-box { height: calc(100% - 95px); margin-top: 10px; overflow: auto; } .item { display: flex; justify-content: space-between; padding: 5px 10px; font-size: 14px; color: #606266; cursor: pointer; border-radius: 4px; &.active { background: #e6f5f3 !important; } &:hover { background: rgb(245 247 250); } span { margin-left: 10px; cursor: pointer; i { margin-right: 3px; } &:first-of-type { color: #009688; } &:last-of-type { color: #f56c6c; } } } .empty { width: 100%; height: 100%; .el-empty { transform: translate(0, 50%); } } </style> 报错如何解决useTable.ts:64 [Vue warn]: Invalid prop: type check failed for prop "data". Expected Array, got Object at <ElTable ref="tableRef" data= {data: Array(1), total: 1, pageNum: 1, pageSize: 20}data: [{…}]pageNum: 1pageSize: 20total: 1[[Prototype]]: Object border=true ... > at ... > at <StandardProductFieldManage onVnodeUnmounted=fn<onVnodeUnmounted> ref=Ref< Proxy(Object) {__v_skip: true} > > at </system/standardProductFieldManage key="/system/standardProductFieldManage" > at <KeepAlive include= (2) ['/system/supplier', '/home/index'] > at <BaseTransition mode="out-in" appear=true persisted=false ... > at <Transition appear="" name="fade-transform" mode="out-in" > at <RouterView> at <ElMain> at <Index> at <ElContainer> at <ElContainer class="layout" > at <LayoutVertical> at <Layout onVnodeUnmounted=fn<onVnodeUnmounted> ref=Ref< Proxy(Object) {__v_skip: true} > > at <RouterView> at <ElConfigProvider size="default" button= {autoInsertSpace: false} > at <App> warn$1 @ chunk-UQWBJQZ5.js?v=58f10a0a:2116 validateProp @ chunk-UQWBJQZ5.js?v=58f10a0a:6468 validateProps @ chunk-UQWBJQZ5.js?v=58f10a0a:6440 updateProps @ chunk-UQWBJQZ5.js?v=58f10a0a:6237 updateComponentPreRender @ chunk-UQWBJQZ5.js?v=58f10a0a:7545 componentUpdateFn @ chunk-UQWBJQZ5.js?v=58f10a0a:7467 run @ chunk-UQWBJQZ5.js?v=58f10a0a:481 updateComponent @ chunk-UQWBJQZ5.js?v=58f10a0a:7342 processComponent @ chunk-UQWBJQZ5.js?v=58f10a0a:7277 patch @ chunk-UQWBJQZ5.js?v=58f10a0a:6782 patchBlockChildren @ chunk-UQWBJQZ5.js?v=58f10a0a:7136 processFragment @ chunk-UQWBJQZ5.js?v=58f10a0a:7214 patch @ chunk-UQWBJQZ5.js?v=58f10a0a:6756 componentUpdateFn @ chunk-UQWBJQZ5.js?v=58f10a0a:7490 run @ chunk-UQWBJQZ5.js?v=58f10a0a:481 runIfDirty @ chunk-UQWBJQZ5.js?v=58f10a0a:519 callWithErrorHandling @ chunk-UQWBJQZ5.js?v=58f10a0a:2263 flushJobs @ chunk-UQWBJQZ5.js?v=58f10a0a:2471 Promise.then queueFlush @ chunk-UQWBJQZ5.js?v=58f10a0a:2385 queueJob @ chunk-UQWBJQZ5.js?v=58f10a0a:2380 effect2.scheduler @ chunk-UQWBJQZ5.js?v=58f10a0a:7532 trigger @ chunk-UQWBJQZ5.js?v=58f10a0a:509 endBatch @ chunk-UQWBJQZ5.js?v=58f10a0a:567 trigger @ chunk-UQWBJQZ5.js?v=58f10a0a:954 set @ chunk-UQWBJQZ5.js?v=58f10a0a:1235 getTableList @ useTable.ts:64 await in getTableList (anonymous) @ index.vue:184 (anonymous) @ chunk-UQWBJQZ5.js?v=58f10a0a:4901 callWithErrorHandling @ chunk-UQWBJQZ5.js?v=58f10a0a:2263 callWithAsyncErrorHandling @ chunk-UQWBJQZ5.js?v=58f10a0a:2270 hook.__weh.hook.__weh @ chunk-UQWBJQZ5.js?v=58f10a0a:4881 flushPostFlushCbs @ chunk-UQWBJQZ5.js?v=58f10a0a:2448 flushJobs @ chunk-UQWBJQZ5.js?v=58f10a0a:2490 Promise.then queueFlush @ chunk-UQWBJQZ5.js?v=58f10a0a:2385 queueJob @ chunk-UQWBJQZ5.js?v=58f10a0a:2380 effect2.scheduler @ chunk-UQWBJQZ5.js?v=58f10a0a:7532 trigger @ chunk-UQWBJQZ5.js?v=58f10a0a:509 endBatch @ chunk-UQWBJQZ5.js?v=58f10a0a:567 notify @ chunk-UQWBJQZ5.js?v=58f10a0a:827 trigger @ chunk-UQWBJQZ5.js?v=58f10a0a:801 set value @ chunk-UQWBJQZ5.js?v=58f10a0a:1673 handleItemClick @ index.vue:97 onClick @ index.vue:9 callWithErrorHandling @ chunk-UQWBJQZ5.js?v=58f10a0a:2263 callWithAsyncErrorHandling @ chunk-UQWBJQZ5.js?v=58f10a0a:2270 invoker @ chunk-UQWBJQZ5.js?v=58f10a0a:11202 useTable.ts:64 [Vue warn]: Unhandled error during execution of app errorHandler at <ElTable ref="tableRef" data= {data: Array(1), total: 1, pageNum: 1, pageSize: 20} border=true ... > at ... > at <StandardProductFieldManage onVnodeUnmounted=fn<onVnodeUnmounted> ref=Ref< Proxy(Object) {__v_skip: true} > > at </system/standardProductFieldManage key="/system/standardProductFieldManage" > at <KeepAlive include= (2) ['/system/supplier', '/home/index'] > at <BaseTransition mode="out-in" appear=true persisted=false ... > at <Transition appear="" name="fade-transform" mode="out-in" > at <RouterView> at <ElMain> at <Index> at <ElContainer> at <ElContainer class="layout" > at <LayoutVertical> at <Layout onVnodeUnmounted=fn<onVnodeUnmounted> ref=Ref< Proxy(Object) {__v_skip: true} > > at <RouterView> at <ElConfigProvider size="default" button= {autoInsertSpace: false} > at <App>

<script setup lang="ts"> import { onMounted, ref,watch } from "vue"; import { $t } from "@/locales"; import { computed } from 'vue'; import { sourceOptions, softwareChangeOptions, canUpgradeOptions } from "@/constants/business"; import { translateOptions } from "@/utils/common"; import { useNaiveForm } from "@/hooks/common/form";; import { router } from "@/router"; //import {getDeptOptions} from "@/service/api"; import {useDepartment} from "@/hooks/common/useDepartment" import { NInput } from "naive-ui"; // 使用封装的部门功能 const { deptLoading, filteredDeptOptions, handleDeptSearch, resetDeptSearch } = useDepartment(); const currentRoute = ref(router.currentRoute); defineOptions({ name: "LedgerSearch", }); interface Emits { (e: "reset"): void; (e: "search"): void; } const emit = defineEmits<Emits>(); const { formRef, validate, restoreValidation } = useNaiveForm(); const model = defineModel<Api.Change.LedgerSearchParams>("model", { required: true, }); async function reset() { await restoreValidation(); resetDeptSearch(); // 使用封装的reset函数 emit("reset"); } async function search() { await validate(); emit("search"); } onMounted(async () => { }); </script> <template> <NCard :bordered="false" size="small" class="card-wrapper"> <NCollapse> <NCollapseItem :title="$t('common.search')" name="change-ledger-search"> <NForm ref="formRef" :model="model" label-placement="left" :label-width="80" > <NGrid responsive="screen" item-responsive> <NFormItemGi span="24 s:12 m:6" :label="$t('page.change.ledger.source')" path="source" class="pr-24px" > <NSelect v-model:value="model.source" :placeholder="$t('page.change.ledger.form.source')" :options="translateOptions(sourceOptions)" filterable clearable /> </NFormItemGi> <NFormItemGi span="24 s:12 m:6" :label="$t('page.change.ledger.peco')" path="peco" class="pr-24px" > <NInput v-model:value="model.peco" :placeholder="$t('page.change.ledger.form.peco')" /> </NFormItemGi> <NFormItemGi span="24 s:12 m:6" :label="$t('page.change.ledger.projectno')" path="projectno" class="pr-24px" > <NInput v-model:value="model.projectno" :placeholder="$t('page.change.ledger.form.projectno')" /> </NFormItemGi> <NFormItemGi span="24 s:12 m:6" :label="$t('page.change.ledger.dept')" path="dept" class="pr-24px" > <NSelect v-model:value="model.dept" :placeholder="$t('page.change.ledger.form.dept')" :options="filteredDeptOptions" :loading="deptLoading" filterable clearable @search="handleDeptSearch" /> </NFormItemGi> <NFormItemGi span="24 s:12 m:6" :label="$t('page.change.ledger.canUpgrade')" path="canUpgrade" class="pr-24px" > <NSelect v-model:value="model.canUpgrade" :placeholder="$t('page.change.ledger.form.canUpgrade')" :options="translateOptions(canUpgradeOptions)" filterable clearable /> </NFormItemGi> <NFormItemGi span="24 s:12 m:6" :label="$t('page.change.ledger.softwareChange')" path="softwareChange" class="pr-24px" > <NInput v-model:value="model.softwareChange" :placeholder="$t('page.change.ledger.form.softwareChange')" /> </NFormItemGi> <NFormItemGi span="24 s:12 m:6"> <NSpace class="w-full" justify="end"> <NButton @click="reset"> <template #icon> <icon-ic-round-refresh class="text-icon" /> </template> {{ $t("common.reset") }} </NButton> <NButton type="primary" ghost @click="search"> <template #icon> <icon-ic-round-search class="text-icon" /> </template> {{ $t("common.search") }} </NButton> </NSpace> </NFormItemGi> </NGrid> </NForm> </NCollapseItem> </NCollapse> </NCard> </template> <style scoped></style> 在这里进行使用

最新推荐

recommend-type

securitylake-jvm-1.0.47.jar

securitylake-jvm-1.0.47.jar
recommend-type

Flash动画制作基础与应用:从操作到企业宣传动画实战

资源摘要信息: Flash动画制作作为模块四的核心内容,涵盖了从基础操作到高级动画制作技巧的完整教学体系。本教学文档主要分为四个任务模块,分别对应不同的学习阶段和实践目标,旨在通过“教、学、做一体”的教学方法,使学习者逐步掌握Flash动画制作的核心技能。 任务一主要聚焦于Flash软件的基本操作和简单动画效果的实现。教学目标明确指出,学生需要认识Flash软件的功能与基本操作界面,包括菜单栏、常用工具栏、绘图工具栏、图层区、时间轴以及各类设置面板。此外,还需要了解库、元件、场景、帧、关键帧等核心概念。这些概念构成了Flash动画制作的理论基础:库是存储元件的容器,元件是可重复使用的对象,场景是动画展示的舞台,时间轴则负责组织和控制动画的时间流程,而帧和关键帧则是构成动画时间线的基本单位。教学内容涵盖了文档属性的设置、外部对象的导入方法、基本图形的绘制技巧,以及简单的动作动画制作,最终还包括文件的保存与影片的导出流程。通过任务一的学习,学生可以初步掌握如何在Flash中构建动画的基本框架。 任务二深入讲解了动画制作的多种类型与实现方式,具体包括逐帧动画、运动渐变动画、形状补间动画以及遮罩效果的制作。逐帧动画是最基础的动画形式,通过连续播放多个关键帧来实现动画效果,适用于细节要求较高的动画场景;而运动渐变动画和形状补间动画则属于补间动画的范畴,只需设置起始和结束关键帧,中间的变化由软件自动计算完成,大大提升了动画制作的效率。遮罩效果则是一种高级动画技巧,常用于制作动态遮挡、图像切换等视觉效果,增强了动画的表现力。通过任务二的实践操作,学生将掌握多种动画类型的制作方法,并能够根据实际需求选择合适的动画技术进行创作。 任务三与任务四则侧重于实际项目的应用,分别聚焦于企业宣传动画和商品推广动画的制作。教学设计中强调了动画在宣传与展示中的重要作用,例如提升视觉吸引力、传递企业文化和推广商品等。在任务三中,学生将学习如何将企业理念、产品特色与动画艺术结合,通过动画的形式进行企业形象的宣传。任务四则进一步拓展至电商领域,引导学生掌握如何制作具有营销效果的商品推广动画,包括如何将商品特性通过动画形式生动呈现,吸引潜在消费者的注意力。这两个任务不仅强调技术实现,更注重创意表达与视觉传达能力的培养,使学生能够综合运用所学知识,完成具有实际应用价值的动画作品。 在课程引入部分,详细阐述了网页的构成元素,如文本、表格、图片、图表和动画等,并解释了动画为何能够“动”的原理——即通过连续播放一系列基于时间顺序的静态画面,利用人眼的视觉暂留现象,形成连续变化的动态效果。教学内容还比较了二维动画与三维动画的区别,以及逐帧动画与补间动画的技术特点,帮助学生理解不同动画形式的应用场景。同时,动画的作用也被明确指出,包括美观直观的视觉呈现、有效的宣传展示功能以及吸引受众注意力等。 关于Flash软件的功能介绍,文档强调其作为一款功能强大且易于学习的动画制作工具,具备多种优势。例如,使用Flash制作的动画不仅文件体积小,而且画质高,无论放大或缩小均不影响画面清晰度。该软件主要支持补间动画制作,即只需设定起始与结束画面,中间过渡由软件自动处理生成,极大地提高了制作效率。此外,Flash还支持交互功能,用户可通过点击按钮、选择菜单等方式控制动画播放,甚至可以制作成互动小游戏。在网络应用方面,Flash动画支持边下载边播放的功能,适应了网络传输的需求。 Flash的软件界面由多个功能区域组成,包括菜单栏、常用工具栏、绘图工具栏、图层区、时间轴和各类设置面板。文档属性设置涉及画布尺寸、背景颜色等基本参数。元件作为动画中的“演员”,包括图片、文字等可重复使用的对象,统一存储在库中,使用时拖放至场景即可。场景是动画实际呈现的舞台,时间轴则作为动画的“剧本”,通过帧的组织与排列控制动画的播放顺序。通过系统的学习,学生可以掌握Flash软件的完整操作流程,并能够独立完成从构思到实现的动画制作全过程。 综上所述,该教学文档全面覆盖了Flash动画制作的基础知识、核心技术以及实际应用案例,通过任务驱动的教学模式,引导学生逐步掌握动画制作的各项技能,并具备将所学知识应用于企业宣传与商品推广等实际场景的能力。
recommend-type

模型量化深度解析:将浮点模型压缩为8位整数的完整技术路径

# 1. 模型量化的背景与核心价值 随着深度学习模型规模持续膨胀,从BERT到GPT系列,参数量动辄数十亿,传统浮点推理已难以满足边缘设备对延迟、功耗和存储的严苛要求。模型量化应运而生,成为工业界落地AI的关键使能技术。其核心思想是将高精度浮点权重与激活值映射为低比特整数(如INT8),在几乎不损失精度的前提下,显著压缩模型体积、提升推理速度并降低计算能耗。 量化不仅适用于云端大
recommend-type

milvus 向量数据库如何加在collection

### 如何在 Milvus 向量数据库中加载 Collection 在 Milvus 向量数据库中,`Collection` 是数据存储的核心结构之一。为了能够对 `Collection` 执行搜索或其他操作,必须先将其加载到内存中。以下是有关如何加载 `Collection` 的详细说明: #### 1. 加载 Collection 前的准备 在加载 `Collection` 之前,需要确保已经完成了以下准备工作: - 创建了一个有效的 `Collection` 并插入了数据[^2]。 - 安装并配置好了 Milvus Python SDK (`pymilvus`) 或其他支持的语言
recommend-type

我国国际贸易中电子商务的发展现状与挑战分析

资源摘要信息:"电子商务在国际贸易实践中的应用与挑战" 随着信息技术的迅猛发展,电子商务在全球范围内逐渐成为推动国际贸易发展的重要引擎。电子商务作为新型的商业运作模式,不仅改变了传统贸易的交易方式,还深刻影响了全球经济结构和企业运营模式。本文以我国的电子商务在国际贸易中的实际应用为基础,深入探讨了其发展现状、存在问题及未来发展方向,为推动我国电子商务在国际贸易领域的进一步发展提供了理论支持和实践指导。 首先,电子商务在国际贸易中的应用为我国外贸体系注入了新的活力。根据文中引用的北京互联网发展中心(BIDC)在第87届中国出口商品交易会上进行的调查数据,我国出口企业在电子商务的应用方面呈现出显著的增长趋势。调查结果显示,4.5%的出口企业已经广泛开展电子商务,31.2%的企业处于初步应用阶段,28.7%的企业已着手准备开展,另有12.0%的企业已有相关考虑,仅有23.6%的企业尚未涉足。这一数据充分表明,我国出口企业在电子商务领域的发展潜力巨大,越来越多的企业开始意识到电子商务在拓展国际市场、提升交易效率和降低运营成本方面的优势。 阿里巴巴等外贸电商平台的兴起,进一步推动了我国企业参与全球贸易的深度和广度。大量企业在该平台注册并开展外贸业务,配备了专门的外贸业务员进行线上交易操作,这标志着我国外贸企业正逐步向数字化、智能化方向转型。这种转型不仅提高了企业的国际竞争力,也为我国整体外贸体系的升级提供了支撑。 然而,尽管电子商务在我国国际贸易中展现出良好的发展态势,仍存在诸多问题亟待解决。首先,公众对电子商务的认知程度仍有待提高。许多企业尤其是中小型企业对电子商务的理解仍停留在表面阶段,缺乏系统的战略规划和专业人才支持,这在一定程度上限制了其在国际贸易中的有效应用。 其次,国际标准与协议的参与度不足。文中提到,1997年6月,国际标准化组织(ISO/IEC JTC1)成立了电子商务业务工作组(BT-EC),并明确了电子商务标准化的三个重点领域:用户接口、基本功能以及数据与客体的定义与编码。虽然这些标准的制定为全球电子商务的发展提供了基础框架,但我国在国际标准制定中的参与度和影响力相对较低,这在一定程度上影响了我国企业在全球电子商务体系中的话语权。 此外,关键技术的研发与应用滞后也是我国电子商务在国际贸易中面临的重要挑战。当前,信息技术主要被发达国家所垄断,其在技术输出方面设置了诸多壁垒,严格控制核心技术向我国的转移。这种技术垄断现象严重制约了我国企业在电子商务领域的自主创新能力,影响了我国在全球电子商务产业链中的地位提升。 法律与制度环境的不完善也是电子商务在国际贸易中应用受限的重要因素。目前,我国及许多国家尚未在法律层面承认电子文件的法律效力。根据我国及各国票据法的规定,具有法律效力的签字必须由法定负责人亲自手书签署,而电子商务交易中的电子签名难以满足这一要求。此外,还存在因网络系统故障导致的电子文件错误的法律责任归属问题、电子数据与网上信息的证据效力问题、以及电子商务交易中的国际商事仲裁问题等。这些问题的存在,增加了企业在电子商务交易中的法律风险,阻碍了其在国际贸易中的广泛应用。 针对上述问题,文章提出了若干应对策略。首先是提高公众对电子商务的认知水平,通过政策引导和教育培训,增强企业对电子商务战略价值的理解和应用能力;其次是积极参与国际公约和贸易伙伴协议,提升我国在国际电子商务标准制定中的话语权;再次是加大关键技术的研发投入,提升我国在电子商务核心技术领域的自主创新能力;最后是完善电子商务相关法律法规体系,推动电子签名、电子合同等电子文件的法律认可,为电子商务在国际贸易中的健康发展提供坚实的法律保障。 综上所述,电子商务作为现代国际贸易的重要推动力量,其发展不仅关系到企业的国际竞争力,也对我国整体外贸体系的现代化进程具有深远影响。尽管当前我国在电子商务应用于国际贸易的过程中面临诸多挑战,但通过加强公众认知、提升技术研发能力、积极参与国际标准制定以及完善相关法律制度,我国有望在全球电子商务体系中占据更加有利的位置,从而实现外贸领域的高质量发展。
recommend-type

低功耗采样设计艺术:精准平衡精度与能耗的4大原则

# 1. 低功耗采样设计的核心挑战与系统视角 在物联网、可穿戴设备与边缘传感系统中,低功耗采样设计已成为能效优化的核心环节。传统奈奎斯特采样往往造成能量浪费,尤其在信号稀疏或缓变场景下,持续高率采样显著拉高系统功耗。真正的挑战不仅在于降低ADC工作频率,更在于构建一个从信号特性感知、采样策略动态
recommend-type

短作业优先调度算法头哥答案

### 短作业优先调度算法(SJF)概述 短作业优先调度算法(Shortest Job First, SJF)是一种基于作业长度的调度策略,其核心思想是优先处理预计运行时间较短的任务。这种算法可以显著减少平均等待时间,从而提升系统整体性能[^1]。 在数据结构方面,SJF通常依赖于队列或堆栈来管理待处理任务列表。为了实现该算法,需维护一个按预期完成时间排序的任务集合,并动态更新此集合中的元素顺序。以下是SJF的一些关键特性: - **优点**:能够最小化平均周转时间,适合批处理环境下的资源分配。 - **缺点**:可能导致较长任务被无限期延迟(饥饿现象),除非引入额外机制解决公平性问题[
recommend-type

峨眉山大酒店网络订房客人接待与服务实施方案

资源摘要信息:"峨眉山大酒店接待网络订房客人实施方案是一份详尽的管理文档,旨在全面提升网络订房客人的服务体验。该方案涵盖了从客人预订开始,到入住期间及离店后的全流程服务规范,目标是实现“0差评”和提升网络好评率。方案明确了各部门的职责分工,包括前厅部、客房部、后勤保卫部、餐饮部、温泉部、网络销售经理以及质量管理办公室等,确保每个环节都有专人负责并协同配合。同时,方案还详细列出了网络订房的主要来源平台,如携程、艺龙、美团、阿里飞猪、官网及识途网,确保酒店在网络渠道的运营策略具有针对性。此外,方案特别强调了客服班组的人员安排与工作时间,要求客服团队在前台提供全程引领服务,并引导客人进行好评。在接待过程中,客服人员的着装规范、服务态度、语言表达能力以及对酒店产品与峨眉山旅游知识的掌握程度都被明确列出,以确保服务质量的一致性。对于不同网络平台的客人,方案还特别安排了房间分配计划,确保网络订房客户在不同楼栋中获得合适的住宿安排。整体来看,这份方案体现了酒店对网络客源的高度重视,通过标准化、流程化、精细化的服务管理,旨在提升客户满意度,增强品牌影响力,并实现从新用户到忠实客户的转化。" 以下是对该文档知识点的详细说明: 一、网络订房客人接待目标体系 1. **全流程服务覆盖** 该方案将网络订房客人的接待工作划分为三个主要阶段:到店前预订、到店时及入住中的接待、离店后的维护。这种全流程覆盖的服务体系,确保了客户在酒店消费过程中的每一个环节都能获得良好的服务体验,从而提升整体满意度。 2. **客户转化目标** 方案明确提出“努力争取把新用户变为老用户”的目标,体现了酒店在客户关系管理上的战略思维。通过提升服务质量和客户体验,酒店希望实现客户忠诚度的提升,增强回头客比例,从而形成稳定的客户群体。 3. **差评控制与好评提升** “0差评”和“提升网络好评率”是该方案的核心质量指标。这一目标的设定,不仅有助于酒店在网络平台上的口碑建设,也直接影响其在线预订量和市场竞争力。为了达成这一目标,方案在服务流程、员工培训、设施管理等方面都做了详细部署。 二、各部门职责与服务要求 1. **前厅部** 负责网络客人房间的安排工作,是整个接待流程的第一环节。若出现特殊情况,需及时与网络部沟通协调,体现了跨部门协作的重要性。前厅部的高效运作对于提升客户第一印象具有关键作用。 2. **客房部** 负责引领服务及房间设施设备的检查。该部门需确保网络房间的设施设备完好无损,避免因硬件问题导致客户投诉。这一职责体现了酒店对细节管理的高度重视。 3. **后勤保卫部** 为网络客人提供免费停车服务,是一项增强客户满意度的增值服务。在当前竞争激烈的酒店行业中,停车便利性往往是客户选择酒店的重要考量因素之一。 4. **餐饮部** 负责早餐服务,特别是对收餐时间的严格把控,确保了服务的标准化与客户体验的一致性。早餐是酒店服务的重要组成部分,良好的早餐体验有助于提升客户整体满意度。 5. **温泉部** 作为峨眉山大酒店的特色服务之一,温泉服务的安全管理被特别强调。安全是服务行业的基础,尤其在涉及水疗和温泉等具有一定风险的服务中,安全管理尤为重要。 6. **网络销售经理** 负责网络客人的到店引领、问询以及离店后的维护工作。该角色在客户关系管理中扮演着重要角色,是连接线上与线下的关键节点。通过有效的客户维护,可以提升客户复购率和好评率。 7. **质管办(质量管理办公室)** 作为监督部门,负责对各部门服务进行质量检查,确保各项服务标准得以落实。质量管理是酒店运营的核心环节,通过持续监督和改进,能够有效提升整体服务水平。 三、网络订房渠道管理 方案明确指出网络订房的主要来源平台为:携程、艺龙、美团、阿里飞猪、官网、识途网六大网站。这表明酒店在网络营销渠道的布局上具有明确的战略规划,能够根据不同平台的用户特点制定差异化的服务策略。 1. **携程、艺龙等主流平台** 这些平台用户基数大、评价机制成熟,是酒店获取优质客户的重要来源。酒店需在这些平台上保持良好评分和形象,以吸引更多预订。 2. **官网与识途网** 官网是酒店品牌形象的重要窗口,识途网则可能聚焦于特定客户群体。通过官网和识途网的运营,酒店可以实现品牌宣传与客户引流的双重目标。 四、客服班组配置与服务标准 1. **人员组成** 由客服班组石有红等三人组成接待小组,体现出酒店对客服团队专业性的重视。小团队配置有利于提高服务响应速度和客户接待效率。 2. **工作时间安排** 工作时间为8:00-21:00,周末延长至23:00,覆盖了大部分客人到店时间,确保每一位网络订房客人都能获得及时引导和服务。 3. **服务规范要求** 包括着装规范(淡妆、扎发、工号牌佩戴)、服务态度(热情、礼貌)、语言表达(普通话标准)、知识储备(酒店产品与峨眉山旅游知识)。这些细节要求不仅提升了服务的专业性,也增强了客户对酒店品牌的认同感。 五、房间分配策略 针对携程、去哪儿网、艺龙等平台的网络订房客人,方案详细列出了各楼栋的房间安排,包括单间与标间的具体房号。这种分配策略有助于: 1. **提升客户体验** 为网络客人安排特定房间,避免因房间质量问题引发投诉,保障客户入住体验。 2. **管理资源分配** 通过对不同楼栋、楼层的房间进行统一安排,酒店可以更高效地调配资源,提升客房使用率。 3. **增强服务一致性** 通过标准化房间配置,确保每位网络客人获得相似的服务质量,减少因服务差异带来的负面评价。 六、客户引导与网络评价管理 方案中特别强调了客服人员在引导客人办理入住和退房时,要引导客人进行好评。这一举措反映了酒店对网络评价的高度关注,也体现了其对客户反馈管理的重视。 1. **好评引导策略** 通过服务人员的主动引导,提高客户在平台上的评价意愿,有助于提升酒店在网络平台上的评分和曝光率。 2. **客户维护机制** 在离店后继续进行客户维护,包括回访、优惠推送等,有助于增强客户粘性,促进复购。 综上所述,该方案系统性地构建了网络订房客户服务的全流程管理体系,涵盖了目标设定、职责分工、人员配置、服务标准、客户引导等多个方面,体现了峨眉山大酒店在网络时代对客户体验管理的高度重视与专业运作。
recommend-type

ESP32内存优化终极方案:突破TinyML模型加载瓶颈的8种策略

# 1. ESP32内存架构与TinyML模型加载瓶颈解析 ESP32作为主流的边缘AI终端芯片,其双核Xtensa架构和丰富的外设支持使其成为TinyML部署的理想平台。然而,在实际模型加载过程中,开发者常遭遇模型无法加载或运行时崩溃等问题,根源在于对ESP32复杂的内存架构理解不足。本章将深入剖析其内存布局特性与TinyML模型间的资源冲突,揭示模型加载失败的本质原因,为后续优化提供理论切入点。 # 2. 内存优化核心理论基础
recommend-type

snabbdom

### Snabbdom 库简介 Snabbdom 是一个轻量级的 JavaScript 虚拟 DOM 实现库,专注于高性能和灵活性。它提供了一种高效的方式来管理 DOM 更新,类似于 React 的虚拟 DOM 机制[^3]。然而,Snabbdom 更加模块化,允许开发者仅引入所需的功能模块。 #### 核心特性 1. **模块化设计** Snabbdom 使用插件化的架构来扩展功能。核心库非常精简,而额外的功能(如属性处理、事件监听器等)可以通过独立的模块加载。这种设计使得 Snabbdom 非常适合构建自定义框架或工具链[^4]。 2. **高效的 Diff 算法**