SouceModal.tsx 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. import Thumbnail from "@/components/Thumbnail";
  2. import { SelectOneImage } from "@/pages/website/Material2/modal";
  3. import { css } from "@linaria/core";
  4. import { Button, Form, Input, TreeSelect } from "ant-design-vue";
  5. import { useModal } from "queenjs";
  6. import { defineComponent, nextTick, onMounted, reactive, ref } from "vue";
  7. import { any, string } from "vue-types";
  8. import { useResource } from "..";
  9. export default defineComponent({
  10. props: {
  11. item: any(),
  12. sourceType: string().isRequired,
  13. },
  14. setup(props) {
  15. const { controls } = useResource();
  16. let data = props.item
  17. ? { ...props.item }
  18. : {
  19. title: "",
  20. categories: [],
  21. };
  22. if (!data.categories) {
  23. data.categories = [];
  24. }
  25. const formState: { [name: string]: any } = reactive({ ...data });
  26. const sourceTitle = ["comp", "shape", "text", "template", "v"];
  27. const state = reactive({
  28. categoriesVal: {} as any,
  29. categoryList: [] as any,
  30. categoryMap: new Map<string, any>(),
  31. });
  32. const getCate = () => {
  33. const categories = controls.categoryCtrl.state.categories;
  34. const currCate: any = categories.find((e: any) => {
  35. return e.value == props.sourceType;
  36. });
  37. initCateMap(currCate?.children || []);
  38. state.categoryList = currCate?.children || [];
  39. };
  40. const initCateMap = (options: any, parent = "") => {
  41. options.map((e: any) => {
  42. state.categoryMap.set(e.id, {
  43. parent: parent,
  44. children: e.children || undefined,
  45. });
  46. if (e.children) {
  47. initCateMap(e.children, e.id);
  48. }
  49. });
  50. };
  51. onMounted(() => {
  52. getCate();
  53. nextTick(() => {
  54. initCateValue();
  55. });
  56. });
  57. const initCateValue = () => {
  58. let values: any = {};
  59. formState.categories.map((id: string) => {
  60. const rootId = getRootParent(id);
  61. if (rootId) {
  62. values[rootId] || (values[rootId] = []);
  63. values[rootId].push(id);
  64. }
  65. });
  66. state.categoriesVal = values;
  67. };
  68. const getRootParent = (id: string): string => {
  69. const item = state.categoryMap.get(id);
  70. if (!item.parent) {
  71. return id;
  72. } else {
  73. return getRootParent(item.parent);
  74. }
  75. };
  76. const rules = ref({
  77. title: [
  78. {
  79. required: props.sourceType != "image" ? true : false,
  80. message: "名称不能为空!",
  81. },
  82. ],
  83. thumbnail: [
  84. {
  85. required: props.sourceType != "image" ? true : false,
  86. message: "封面图不能为空",
  87. },
  88. ],
  89. });
  90. const { validate, validateInfos } = Form.useForm(formState, rules);
  91. const modal = useModal();
  92. const submit = () => {
  93. validate().then(async (values) => {
  94. let categoriesVal: any = [];
  95. Object.values(state.categoriesVal).forEach((e: any) => {
  96. categoriesVal = [...categoriesVal, ...e];
  97. });
  98. formState.categories = categoriesVal;
  99. modal.submit(formState);
  100. });
  101. };
  102. const changeThumbnail = async () => {
  103. if (props.sourceType == "image") {
  104. return;
  105. }
  106. const url = await SelectOneImage();
  107. if (!url) return;
  108. formState.thumbnail = url;
  109. };
  110. return () => {
  111. const thumbnail =
  112. props.sourceType != "image" ? formState.thumbnail : formState.file.url;
  113. return (
  114. <div class={configFormStyle}>
  115. <div class="modal_form">
  116. <Form name="basic">
  117. <Form.Item name="thumbnail" {...validateInfos.thumbnail}>
  118. <div class={"h-180px bg-light-50"} onClick={changeThumbnail}>
  119. <Thumbnail
  120. src={thumbnail}
  121. class="h-1/1 w-1/1"
  122. objectContain={props.sourceType == "image" ? true : false}
  123. />
  124. </div>
  125. </Form.Item>
  126. {sourceTitle.includes(props.sourceType) && (
  127. <Form.Item name="title" {...validateInfos.title}>
  128. <Input v-model={[formState.title, "value"]} />
  129. </Form.Item>
  130. )}
  131. {state.categoryList.map((item: any, index: number) => {
  132. if (!item.isEdit) return null;
  133. return (
  134. <Form.Item key={index} name={`categories.${index}`}>
  135. <TreeSelect
  136. value={state.categoriesVal[item.id]}
  137. allowClear
  138. treeCheckable
  139. dropdownStyle={{ maxHeight: "500px", overflow: "auto" }}
  140. placeholder={`请选择${item.name}`}
  141. treeData={item?.children}
  142. replaceFields={{
  143. children: "children",
  144. label: "name",
  145. value: "id",
  146. }}
  147. onChange={(v) => {
  148. state.categoriesVal[item.id] = v;
  149. }}
  150. />
  151. </Form.Item>
  152. );
  153. })}
  154. </Form>
  155. </div>
  156. <div class="modal_footer">
  157. <Button onClick={modal.cancel}>取消</Button>
  158. <Button type="primary" onClick={submit}>
  159. 确定
  160. </Button>
  161. </div>
  162. </div>
  163. );
  164. };
  165. },
  166. });
  167. const configFormStyle = css`
  168. .thumb_wapper {
  169. height: 180px;
  170. }
  171. .modal_footer {
  172. display: flex;
  173. justify-content: flex-end;
  174. padding: 10px 0;
  175. .ant-btn + .ant-btn {
  176. margin-left: 10px;
  177. }
  178. }
  179. `;