Notice-edit.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543
  1. <template>
  2. <el-dialog v-model="state.dialogVisible" width="50%" draggable :title="dialogTitle" @close="close">
  3. <el-form :model="state.ruleForm" label-width="110px" ref="ruleFormRef" v-loading="loading">
  4. <template v-if="dialogTitle === '编辑通知'">
  5. <el-row :gutter="10">
  6. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  7. <el-form-item label="通知类型" prop="circularObj" :rules="[{ required: true, message: '请选择通知类型型', trigger: 'change' }]">
  8. <el-select
  9. v-model="state.ruleForm.circularObj"
  10. placeholder="请选择通知类型型"
  11. value-key="dicDataValue"
  12. class="w100"
  13. @change="
  14. (e) => {
  15. state.ruleForm.circularTypeId = e.dicDataValue;
  16. state.ruleForm.circularTypeName = e.dicDataName;
  17. }
  18. "
  19. >
  20. <el-option v-for="item in circularTypeOptions" :value="item" :key="item.dicDataValue" :label="item.dicDataName" />
  21. </el-select>
  22. </el-form-item>
  23. </el-col>
  24. <el-col :xs="24" :sm="24" :md="12" :lg="12" :xl="12">
  25. <el-form-item label="通知标题" prop="title" :rules="[{ required: true, message: '请输入通知标题', trigger: 'blur' }]">
  26. <el-input v-model="state.ruleForm.title" placeholder="请输入通知标题" clearable show-word-limit maxlength="200"></el-input>
  27. </el-form-item>
  28. </el-col>
  29. <el-col :xs="24" :sm="24" :md="12" :lg="12" :xl="12">
  30. <el-form-item label="必须阅读" prop="isMustRead" :rules="[{ required: true, message: '请选择必须阅读', trigger: 'change' }]">
  31. <el-radio-group v-model="state.ruleForm.isMustRead">
  32. <el-radio :label="true">是</el-radio>
  33. <el-radio :label="false">否</el-radio>
  34. </el-radio-group>
  35. </el-form-item>
  36. </el-col>
  37. <el-col :xs="24" :sm="24" :md="12" :lg="12" :xl="12">
  38. <el-form-item label="失效时间" prop="lostEfficacyTime" :rules="[{ required: true, message: '请选择失效时间', trigger: 'change' }]">
  39. <el-date-picker
  40. v-model="state.ruleForm.lostEfficacyTime"
  41. type="datetime"
  42. placeholder="请选择失效时间"
  43. value-format="YYYY-MM-DD[T]HH:mm:ss"
  44. class="w100"
  45. :disabled-date="disabledDate"
  46. popper-class="no-atTheMoment"
  47. />
  48. </el-form-item>
  49. </el-col>
  50. <el-col :xs="24" :sm="24" :md="12" :lg="12" :xl="12">
  51. <el-form-item label="来源单位" prop="sourceOrgId" :rules="[{ required: true, message: '请选择来源单位', trigger: 'change' }]">
  52. <el-cascader
  53. :options="orgsOptions"
  54. filterable
  55. :props="{ value: 'id', label: 'name', emitPath: false, checkStrictly: true }"
  56. placeholder="请选择来源单位"
  57. class="w100"
  58. v-model="state.ruleForm.sourceOrgId"
  59. ref="orgRef"
  60. @change="changeOrg"
  61. >
  62. </el-cascader>
  63. </el-form-item>
  64. </el-col>
  65. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  66. <el-form-item label="通知内容" prop="content" :rules="[{ required: true, message: '请输入通知内容', trigger: 'blur' }]">
  67. <editor v-model:get-html="state.ruleForm.content" placeholder="请输入通知内容" height="350px" />
  68. </el-form-item>
  69. </el-col>
  70. <el-col :xs="24" :sm="24" :md="12" :lg="12" :xl="12">
  71. <el-form-item label="通知对象" prop="circularType" :rules="[{ required: true, message: '请选择通知对象', trigger: 'change' }]">
  72. <el-radio-group v-model="state.ruleForm.circularType">
  73. <el-radio :label="item.key" v-for="item in circularTypeEnum" :key="item.key">{{item.value}}</el-radio>
  74. </el-radio-group>
  75. </el-form-item>
  76. </el-col>
  77. <el-col
  78. :xs="24"
  79. :sm="24"
  80. :md="24"
  81. :lg="24"
  82. :xl="24"
  83. v-if="state.ruleForm.circularType === 1"
  84. style="border: var(--el-border); border-radius: var(--el-border-radius-base)"
  85. class="pd15 mb20"
  86. >
  87. <div class="flex-center-align mb10">
  88. <el-input v-model="state.searchContent" placeholder="通知人姓名/账号" class="w100" clearable>
  89. <template #prefix>
  90. <SvgIcon name="ele-Search" />
  91. </template>
  92. </el-input>
  93. </div>
  94. <div style="height: 400px">
  95. <el-auto-resizer>
  96. <template #default="{ height, width }">
  97. <el-table-v2 :columns="columns" :data="tables" :width="width" :height="height" fixed />
  98. </template>
  99. </el-auto-resizer>
  100. </div>
  101. </el-col>
  102. <el-col
  103. :xs="24"
  104. :sm="24"
  105. :md="24"
  106. :lg="24"
  107. :xl="24"
  108. v-if="state.ruleForm.circularType === 2"
  109. style="border: var(--el-border); border-radius: var(--el-border-radius-base)"
  110. class="pd15 mb20"
  111. >
  112. <el-input v-model="query" placeholder="部门名称" @input="onQueryChanged" clearable class="mb10" />
  113. <el-tree-v2
  114. :data="circularOrgOptions"
  115. :props="{ value: 'id', label: 'name' }"
  116. show-checkbox
  117. :height="400"
  118. ref="treeSelectRef"
  119. :filter-method="filterMethod"
  120. />
  121. </el-col>
  122. </el-row>
  123. </template>
  124. <template v-if="dialogTitle === '编辑公告'">
  125. <el-row :gutter="10">
  126. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  127. <el-form-item label="公告类型" prop="bulletinObj" :rules="[{ required: true, message: '请选择公告类型', trigger: 'change' }]">
  128. <el-select
  129. v-model="state.ruleForm.bulletinObj"
  130. placeholder="请选择公告类型"
  131. value-key="dicDataValue"
  132. class="w100"
  133. @change="
  134. (e) => {
  135. state.ruleForm.bulletinTypeId = e.dicDataValue;
  136. state.ruleForm.bulletinTypeName = e.dicDataName;
  137. }
  138. "
  139. >
  140. <el-option v-for="item in bulletinTypeOptions" :value="item" :key="item.dicDataValue" :label="item.dicDataName" />
  141. </el-select>
  142. </el-form-item>
  143. </el-col>
  144. <el-col :xs="24" :sm="24" :md="12" :lg="12" :xl="12">
  145. <el-form-item label="公告标题" prop="title" :rules="[{ required: true, message: '请输入公告标题', trigger: 'blur' }]">
  146. <el-input v-model="state.ruleForm.title" placeholder="请输入公告标题" clearable show-word-limit maxlength="200"></el-input>
  147. </el-form-item>
  148. </el-col>
  149. <el-col :xs="24" :sm="24" :md="12" :lg="12" :xl="12">
  150. <el-form-item label="失效时间" prop="loseEfficacyTime" :rules="[{ required: true, message: '请选择失效时间', trigger: 'change' }]">
  151. <el-date-picker
  152. v-model="state.ruleForm.loseEfficacyTime"
  153. type="datetime"
  154. placeholder="请选择失效时间"
  155. value-format="YYYY-MM-DD[T]HH:mm:ss"
  156. class="w100"
  157. :disabled-date="disabledDate"
  158. popper-class="no-atTheMoment"
  159. />
  160. </el-form-item>
  161. </el-col>
  162. <el-col :xs="24" :sm="24" :md="12" :lg="12" :xl="12">
  163. <el-form-item label="来源单位" prop="sourceOrgId" :rules="[{ required: true, message: '请选择来源单位', trigger: 'change' }]">
  164. <el-cascader
  165. :options="orgsOptions"
  166. filterable
  167. :props="{ value: 'id', label: 'name', emitPath: false, checkStrictly: true }"
  168. placeholder="请选择来源单位"
  169. class="w100"
  170. v-model="state.ruleForm.sourceOrgId"
  171. ref="orgRef"
  172. @change="changeOrg"
  173. >
  174. </el-cascader>
  175. </el-form-item>
  176. </el-col>
  177. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  178. <el-form-item label="公告范围" prop="pushRangesArray" :rules="[{ required: true, message: '请选择公告范围', trigger: 'change' }]">
  179. <el-checkbox-group v-model="state.ruleForm.pushRangesArray">
  180. <el-checkbox :label="item" v-for="item in pushRangesOptions" :key="item.dicDataValue">{{ item.dicDataName }}</el-checkbox>
  181. </el-checkbox-group>
  182. </el-form-item>
  183. </el-col>
  184. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  185. <el-form-item label="公告内容" prop="content" :rules="[{ required: true, message: '请输入公告内容', trigger: 'blur' }]">
  186. <editor v-model:get-html="state.ruleForm.content" placeholder="请输入公告内容" height="350px" />
  187. </el-form-item>
  188. </el-col>
  189. </el-row>
  190. </template>
  191. </el-form>
  192. <template #footer>
  193. <span class="dialog-footer">
  194. <el-button @click="closeDialog" class="default-button">取 消</el-button>
  195. <el-button type="primary" @click="onSubmit(ruleFormRef)" :loading="loading">确 定</el-button>
  196. </span>
  197. </template>
  198. </el-dialog>
  199. </template>
  200. <script setup lang="tsx" name="noticeDetail">
  201. import { reactive, ref, defineAsyncComponent, computed, FunctionalComponent, unref, nextTick } from 'vue';
  202. import '@wangeditor/editor/dist/css/style.css'; // 引入 css
  203. import {
  204. bulletinAddBaseData,
  205. bulletinDetail,
  206. circularUpdate,
  207. circularAddBaseData,
  208. circularDetail,
  209. getAllUsers,
  210. bulletinUpdate,
  211. } from '/@/api/auxiliary/notice';
  212. import { throttle } from '/@/utils/tools';
  213. import { CheckboxValueType, ElMessage, FormInstance } from 'element-plus';
  214. import { disabledDate } from '/@/utils/constants';
  215. // 引入组件
  216. const Editor = defineAsyncComponent(() => import('/@/components/Editor/index.vue')); // 富文本编辑器
  217. // 定义子组件向父组件传值/事件
  218. const emit = defineEmits(['updateList']);
  219. const props = defineProps({
  220. commonType: {
  221. type: Array,
  222. default: () => [],
  223. },
  224. });
  225. // 定义变量内容
  226. const state = reactive<any>({
  227. dialogVisible: false,
  228. loading: false,
  229. ruleForm: {
  230. isMustRead: false, // 是否必须阅读
  231. },
  232. searchContent: '', // 搜索内容
  233. });
  234. let loading = ref<boolean>(false); // 加载状态
  235. // 打开弹窗
  236. const ruleFormRef = ref<RefType>();
  237. const dialogTitle = ref<string>('编辑通知');
  238. const bulletinTypeOptions = ref<EmptyArrayType>([]); // 公告类型
  239. const circularTypeOptions = ref<EmptyArrayType>([]); // 通知类型
  240. const circularTypeEnum = ref<EmptyArrayType>([]); // 通知对象
  241. const pushRangesOptions = ref<EmptyArrayType>([]); // 公告范围
  242. const orgsOptions = ref<EmptyArrayType>([]); // 来源单位
  243. const circularOrgOptions = ref<EmptyArrayType>([]); // 通知对象 部门
  244. const userTables = ref<EmptyArrayType>([]); // 通知对象 个人
  245. const openDialog = async (row: any, type: string) => {
  246. dialogTitle.value = type;
  247. loading.value = true;
  248. state.dialogVisible = true;
  249. switch (type) {
  250. case '编辑通知':
  251. try {
  252. const [circularResponse, userResponse, circularDetailResponse] = await Promise.all([
  253. circularAddBaseData(),
  254. getAllUsers(),
  255. circularDetail(row.id),
  256. ]);
  257. circularTypeOptions.value = circularResponse.result?.circularType ?? [];
  258. orgsOptions.value = circularResponse.result?.orgsOptions ?? [];
  259. circularTypeEnum.value = circularResponse.result?.circularTypeEnum ?? [];
  260. circularOrgOptions.value = circularResponse.result?.orgsOptions ?? [];
  261. userTables.value = userResponse.result ?? [];
  262. state.ruleForm = circularDetailResponse.result ?? {};
  263. state.ruleForm.circularObj = {
  264. dicDataValue: circularDetailResponse.result?.circularTypeId,
  265. dicDataName: circularDetailResponse.result?.circularTypeName,
  266. };
  267. if (state.ruleForm.circularType === 1) {
  268. // 个人
  269. for (let i of state.ruleForm.circularReadGroups) {
  270. for (let j of userTables.value) {
  271. if (i.userId === j.id) {
  272. j.checked = true;
  273. }
  274. }
  275. }
  276. } else if (state.ruleForm.circularType === 2) {
  277. // 部门
  278. await nextTick();
  279. const keys = state.ruleForm.circularReadGroups.map((item: any) => {
  280. return item.orgId;
  281. });
  282. treeSelectRef.value.setExpandedKeys(keys);
  283. setTimeout(() => {
  284. treeSelectRef.value.setCheckedKeys(keys);
  285. }, 100);
  286. }
  287. loading.value = false;
  288. } catch (error) {
  289. console.log(error);
  290. loading.value = false;
  291. }
  292. break;
  293. case '编辑公告':
  294. try {
  295. const [responseAnnounce, bulletinDetailResponse] = await Promise.all([bulletinAddBaseData(), bulletinDetail(row.id)]);
  296. bulletinTypeOptions.value = responseAnnounce.result?.bulletinType ?? [];
  297. pushRangesOptions.value = responseAnnounce.result?.pushRanges ?? [];
  298. orgsOptions.value = responseAnnounce.result?.orgsOptions ?? [];
  299. state.ruleForm = bulletinDetailResponse.result ?? {};
  300. state.ruleForm.bulletinObj = {
  301. dicDataValue: bulletinDetailResponse.result?.bulletinTypeId,
  302. dicDataName: bulletinDetailResponse.result?.bulletinTypeName,
  303. };
  304. state.ruleForm.pushRangesArray = [];
  305. const arrr = state.ruleForm.pushRanges.map((item: any) => {
  306. return item.key;
  307. });
  308. for (let i of arrr) {
  309. for (let j of pushRangesOptions.value) {
  310. if (i === j.dicDataValue) {
  311. state.ruleForm.pushRangesArray.push(j);
  312. }
  313. }
  314. }
  315. loading.value = false;
  316. } catch (error) {
  317. console.log(error);
  318. loading.value = true;
  319. }
  320. break;
  321. }
  322. };
  323. const orgRef = ref<RefType>(); //来源单位
  324. const changeOrg = (type: string) => {
  325. const currentNode = orgRef.value.getCheckedNodes();
  326. state[type].sourceOrgName = currentNode[0].label;
  327. };
  328. type SelectionCellProps = {
  329. value: boolean;
  330. intermediate?: boolean;
  331. onChange: (value: CheckboxValueType) => void;
  332. };
  333. const SelectionCell: FunctionalComponent<SelectionCellProps> = ({ value, intermediate = false, onChange }) => {
  334. return <ElCheckbox onChange={onChange} modelValue={value} indeterminate={intermediate} />;
  335. };
  336. const tables = computed({
  337. get() {
  338. // 注意: filter() 不会改变原始数组。
  339. // 模糊搜索
  340. if (!state.searchContent) return userTables.value;
  341. // filter() 方法创建一个新的数组,新数组中的元素是通过检查指定数组中符合条件的所有元素。
  342. // 注意: filter() 不会对空数组进行检测。
  343. return userTables.value.filter((data: any) => {
  344. // some() 方法用于检测数组中的元素是否满足指定条件;
  345. // some() 方法会依次执行数组的每个元素:
  346. // 如果有一个元素满足条件,则表达式返回true , 剩余的元素不会再执行检测;
  347. // 如果没有满足条件的元素,则返回false。
  348. // 注意: some() 不会对空数组进行检测。
  349. // 注意: some() 不会改变原始数组。
  350. return Object.keys(data).some((key) => {
  351. // indexOf() 返回某个指定的字符在某个字符串中首次出现的位置,如果没有找到就返回-1;
  352. // 该方法对大小写敏感!所以之前需要toLowerCase()方法将所有查询到内容变为小写。
  353. return String(data[key]).toLowerCase().indexOf(state.searchContent) > -1;
  354. });
  355. });
  356. },
  357. set(val) {
  358. // userTables.value = val;
  359. },
  360. });
  361. const columns = [
  362. {
  363. key: 'selection',
  364. width: 50,
  365. align: 'center',
  366. cellRenderer: ({ rowData }) => {
  367. const onChange = (value: CheckboxValueType) => (rowData.checked = value);
  368. return <SelectionCell value={rowData.checked} onChange={onChange} />;
  369. },
  370. headerCellRenderer: () => {
  371. const _data = unref(tables);
  372. const onChange = (value: CheckboxValueType) =>
  373. (tables.value = _data.map((row) => {
  374. row.checked = value;
  375. return row;
  376. }));
  377. const allSelected = _data.every((row) => row.checked);
  378. const containsChecked = _data.some((row) => row.checked);
  379. return <SelectionCell value={allSelected} intermediate={containsChecked && !allSelected} onChange={onChange} />;
  380. },
  381. },
  382. {
  383. key: 'name',
  384. dataKey: 'name',
  385. title: '姓名',
  386. width: 300,
  387. align: 'center',
  388. },
  389. {
  390. key: 'name',
  391. dataKey: 'userName',
  392. title: '账号',
  393. width: 150,
  394. align: 'center',
  395. },
  396. {
  397. key: 'organization.name',
  398. dataKey: 'organization.name',
  399. title: '所属部门',
  400. width: 150,
  401. align: 'center',
  402. },
  403. {
  404. key: 'staffNo',
  405. dataKey: 'staffNo',
  406. title: '工号',
  407. width: 100,
  408. align: 'center',
  409. },
  410. {
  411. key: 'organization.orgTypeText',
  412. dataKey: 'organization.orgTypeText',
  413. title: '部门类别',
  414. width: 100,
  415. align: 'center',
  416. },
  417. ];
  418. const query = ref('');
  419. // 选择部门
  420. const treeSelectRef = ref<RefType>();
  421. const onQueryChanged = (query: string) => {
  422. treeSelectRef.value!.filter(query);
  423. };
  424. const filterMethod = (query: string, node: any) => {
  425. return node.name!.includes(query);
  426. };
  427. // 保存
  428. const onSubmit = throttle(async (formEl: FormInstance | undefined) => {
  429. if (!formEl) return;
  430. await formEl.validate((valid: boolean) => {
  431. if (!valid) return;
  432. loading.value = true;
  433. switch (dialogTitle.value) {
  434. case '编辑通知':
  435. const checkedTable = tables.value?.filter((item: any) => item.checked).map((item: any) => item);
  436. const nodes = treeSelectRef.value?.getCheckedNodes();
  437. if (state.ruleForm.circularType === 1 && !checkedTable.length) {
  438. ElMessage.warning('请选择通知人');
  439. loading.value = false;
  440. return;
  441. }
  442. if (state.ruleForm.circularType === 2 && !nodes.length) {
  443. ElMessage.warning('请选择通知部门');
  444. loading.value = false;
  445. return;
  446. }
  447. let circularReadGroups: { userId: any; userName: any }[] | { orgId: any; orgName: any }[] = [];
  448. if (state.ruleForm.circularType === 1) {
  449. //个人
  450. circularReadGroups = checkedTable.map((item: any) => {
  451. return {
  452. userId: item.id,
  453. userName: item.name,
  454. };
  455. });
  456. } else if (state.ruleForm.circularType === 2) {
  457. // 部门
  458. circularReadGroups = nodes.map((item: any) => {
  459. return {
  460. orgId: item.id,
  461. orgName: item.name,
  462. };
  463. });
  464. }
  465. const circularRequest = {
  466. id: state.ruleForm.id,
  467. title: state.ruleForm.title,
  468. content: state.ruleForm.content,
  469. circularTypeId: state.ruleForm.circularTypeId,
  470. circularTypeName: state.ruleForm.circularTypeName,
  471. circularType: state.ruleForm.circularType,
  472. isMustRead: state.ruleForm.isMustRead,
  473. lostEfficacyTime: state.ruleForm.lostEfficacyTime,
  474. sourceOrgId: state.ruleForm.sourceOrgId,
  475. sourceOrgName: state.ruleForm.sourceOrgName,
  476. circularReadGroups,
  477. };
  478. circularUpdate(circularRequest)
  479. .then(() => {
  480. ElMessage.success('操作成功');
  481. emit('updateList');
  482. closeDialog();
  483. loading.value = false;
  484. })
  485. .catch((err: any) => {
  486. console.log(err);
  487. loading.value = false;
  488. });
  489. break;
  490. case '编辑公告':
  491. const pushRanges = state.ruleForm.pushRangesArray.map((item: any) => {
  492. return {
  493. key: item.dicDataValue,
  494. value: item.dicDataName,
  495. };
  496. });
  497. const bulletinRequest = {
  498. id: state.ruleForm.id,
  499. title: state.ruleForm.title,
  500. content: state.ruleForm.content,
  501. bulletinTypeId: state.ruleForm.bulletinTypeId,
  502. bulletinTypeName: state.ruleForm.bulletinTypeName,
  503. loseEfficacyTime: state.ruleForm.loseEfficacyTime,
  504. sourceOrgId: state.ruleForm.sourceOrgId,
  505. sourceOrgName: state.ruleForm.sourceOrgName,
  506. pushRanges,
  507. };
  508. bulletinUpdate(bulletinRequest)
  509. .then(() => {
  510. ElMessage.success('操作成功');
  511. emit('updateList');
  512. closeDialog();
  513. loading.value = false;
  514. })
  515. .catch((err: any) => {
  516. console.log(err);
  517. loading.value = false;
  518. });
  519. break;
  520. }
  521. });
  522. }, 300);
  523. // 关闭弹窗
  524. const closeDialog = () => {
  525. state.dialogVisible = false;
  526. };
  527. const close = () => {
  528. ruleFormRef.value?.clearValidate();
  529. ruleFormRef.value?.resetFields();
  530. };
  531. // 暴露变量
  532. defineExpose({
  533. openDialog,
  534. closeDialog,
  535. });
  536. </script>