index.vue 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318
  1. <script setup lang="ts">
  2. import { ref, onMounted, computed, nextTick } from "vue";
  3. import {v4 as uuid4} from 'uuid'
  4. import {
  5. S3Client,
  6. PutObjectCommand,
  7. ListObjectsCommand,
  8. GetObjectCommand,
  9. DeleteObjectsCommand,
  10. } from "@aws-sdk/client-s3";
  11. import {
  12. Plus,
  13. Refresh,
  14. } from "@element-plus/icons-vue";
  15. import Node from "./Node.vue";
  16. import AppendModal from "./AppendModal.vue";
  17. import type { TreeData } from "./Tree";
  18. import { ElNotification, type ElTree } from "element-plus";
  19. import { MdEditor } from "md-editor-v3";
  20. import { s3ContentsToTree, getTreeLeafs } from "@/utils/s3Helper";
  21. import "md-editor-v3/lib/style.css";
  22. import type ETNode from 'element-plus/es/components/tree/src/model/node'
  23. import type { DragEvents } from 'element-plus/es/components/tree/src/model/useDragNode'
  24. import type {
  25. AllowDropType,
  26. NodeDropType,
  27. } from 'element-plus/es/components/tree/src/tree.type'
  28. const s3 = new S3Client({
  29. credentials: {
  30. accessKeyId: import.meta.env.VITE_AWS_S3_ACCESS_KEY_ID,
  31. secretAccessKey: import.meta.env.VITE_AWS_S3_SECRET_ACCESS_KEY,
  32. },
  33. region: import.meta.env.VITE_AWS_S3_REGION,
  34. });
  35. const tree$ = ref<InstanceType<typeof ElTree>>();
  36. const langSelect = ref('zh-CN')
  37. const langKeyPrefix = computed(() => {
  38. return {
  39. 'zh-CN': 'docs/',
  40. 'zh-HK': 'zh-HK/docs/',
  41. 'en-US': 'en-US/docs/',
  42. }[langSelect.value]
  43. })
  44. const langDataSource = computed<TreeData[]>(() => {
  45. const filtered = dataSource.value.filter(cont => cont.Key.startsWith(langKeyPrefix.value)).map(cont => ({...cont, Key: cont.Key.replace(new RegExp(`^${langKeyPrefix.value}`), '')}))
  46. return s3ContentsToTree(filtered, {}, (r, label, i, a, thisContent) => ({
  47. isDir: a.length !== i + 1,
  48. ...( a.length === i + 1 ? thisContent : {} )
  49. }));
  50. })
  51. const dataSource = ref([]);
  52. const sideLoading = ref(false);
  53. const loadS3DocsListObjects = async () => {
  54. const command = new ListObjectsCommand({ Bucket: import.meta.env.VITE_DOCS_LIST_BUCKET });
  55. const result = await s3.send(command);
  56. return result.Contents!;
  57. };
  58. const loadSideBar = async () => {
  59. sideLoading.value = true;
  60. dataSource.value = await loadS3DocsListObjects();
  61. sideLoading.value = false;
  62. };
  63. onMounted(() => {
  64. loadSideBar();
  65. });
  66. const currentOpenData = ref<TreeData>();
  67. const onNodeClick = async (data: TreeData, node) => {
  68. if (data.isDir) {
  69. return;
  70. }
  71. textLoading.value = true;
  72. const command = new GetObjectCommand({
  73. Bucket: import.meta.env.VITE_DOCS_LIST_BUCKET,
  74. Key: `${langKeyPrefix.value}${data.key}`,
  75. ResponseCacheControl: "no-cache",
  76. });
  77. const file = await s3.send(command);
  78. text.value = await file.Body?.transformToString()!;
  79. currentOpenData.value = data;
  80. textLoading.value = false;
  81. };
  82. const text = ref("");
  83. const textLoading = ref(false);
  84. const onUploadImg = async (files: File[], callback: (urls: string[] | { url: string; alt: string; title: string }[]) => void) => {
  85. let result = []
  86. for (const file of files) {
  87. try {
  88. const key = `${uuid4()}::${file.name}`
  89. const command = new PutObjectCommand({
  90. Bucket: import.meta.env.DOCS_MEDIA_BUCKET,
  91. Key: key,
  92. Body: file,
  93. ACL: 'public-read',
  94. });
  95. const res = await s3.send(command);
  96. result.push({url: `https://${import.meta.env.DOCS_MEDIA_BUCKET}.s3.amazonaws.com/${key}`, alt: file.name, title: file.name})
  97. } catch (e) {
  98. console.error(e)
  99. ElNotification.error(`${file.name} 上传失败`)
  100. }
  101. }
  102. callback(result)
  103. };
  104. const onSave = async () => {
  105. if (!currentOpenData.value) {
  106. ElNotification.info('请先选择文件')
  107. return;
  108. }
  109. textLoading.value = true
  110. try {
  111. const command = new PutObjectCommand({
  112. Bucket: import.meta.env.VITE_DOCS_LIST_BUCKET,
  113. Key: `${langKeyPrefix.value}${currentOpenData.value?.key}`,
  114. Body: text.value,
  115. });
  116. await s3.send(command);
  117. ElNotification.success(`${currentOpenData.value?.key} 保存成功`)
  118. } catch (e) {
  119. console.error(e)
  120. ElNotification.error(`${currentOpenData.value?.key} 保存失败`)
  121. } finally {
  122. textLoading.value = false
  123. }
  124. };
  125. let resolveAppend: Function | undefined;
  126. let rejectAppend: Function | undefined;
  127. const showAppendModal = ref(false);
  128. const appendLoading = ref(false)
  129. const appendContextData = ref()
  130. const onAppend = async (data: TreeData) => {
  131. const modalConfirmPromise = new Promise((resolve, reject) => {
  132. resolveAppend = resolve;
  133. rejectAppend = reject;
  134. });
  135. appendContextData.value = data
  136. showAppendModal.value = true;
  137. let filename, isDir
  138. try {
  139. ( { filename, isDir } = await modalConfirmPromise);
  140. } catch (e) {
  141. return
  142. }
  143. try {
  144. appendLoading.value = true
  145. const key = `${langKeyPrefix.value}${data.key ? `${data.key}/` : ''}${filename}`
  146. if (!isDir) {
  147. const command = new PutObjectCommand({
  148. Bucket: import.meta.env.VITE_DOCS_LIST_BUCKET,
  149. Key: key,
  150. Body: "",
  151. });
  152. await s3.send(command);
  153. }
  154. const newChild: TreeData = {
  155. key,
  156. label: filename,
  157. children: [],
  158. isDir,
  159. };
  160. if (!data.children) {
  161. data.children = [];
  162. }
  163. data.children.push(newChild);
  164. if (!data.key) {
  165. loadSideBar()
  166. }
  167. } catch (e) {
  168. console.error(e);
  169. ElNotification.error('添加失败')
  170. } finally {
  171. appendLoading.value = false
  172. showAppendModal.value = false;
  173. }
  174. };
  175. const onRemove = async (node: ETNode, data: TreeData) => {
  176. const parent = node.parent;
  177. const children = parent.data.children || parent.data;
  178. const index = children.findIndex((d) => d.key === data.key);
  179. sideLoading.value = true
  180. try {
  181. const leafDatas = getTreeLeafs(data)
  182. const command = new DeleteObjectsCommand({
  183. Bucket: import.meta.env.VITE_DOCS_LIST_BUCKET,
  184. Delete: {
  185. Objects: leafDatas.map(d => ({Key: `${langKeyPrefix.value}${d.key}`})),
  186. Quiet: false,
  187. }
  188. })
  189. await s3.send(command)
  190. children.splice(index, 1);
  191. if (parent.level === 0) {
  192. nextTick(loadSideBar)
  193. }
  194. } catch (e) {
  195. console.error(e)
  196. ElNotification.error('删除失败')
  197. } finally {
  198. sideLoading.value = false
  199. }
  200. };
  201. const onDragEnd = (
  202. draggingNode: ETNode,
  203. dropNode: ETNode,
  204. dropType: NodeDropType,
  205. ev: DragEvents
  206. ) => {
  207. sideLoading.value = true
  208. console.log('tree drag end:', draggingNode, dropNode, dropType)
  209. // TODO copy object
  210. // TODO save redis
  211. sideLoading.value = false
  212. }
  213. const allowDrop = (draggingNode: Node, dropNode: Node, type: AllowDropType) => {
  214. return !( !dropNode.data.isDir && type === 'inner' )
  215. }
  216. </script>
  217. <template>
  218. <div class="md-container">
  219. <div class="left" v-loading="sideLoading">
  220. <div class="toolbar">
  221. <el-radio-group v-model="langSelect" size="small">
  222. <el-radio-button label="简体中文" value="zh-CN" />
  223. <el-radio-button label="繁体中文" value="zh-HK" />
  224. <el-radio-button label="English" value="en-US" />
  225. </el-radio-group>
  226. <el-button-group>
  227. <el-button
  228. @click="onAppend({ key: '', label: '', children: langDataSource })"
  229. type="primary"
  230. text
  231. :icon="Plus"
  232. >
  233. </el-button>
  234. <el-button @click="loadSideBar" type="info" text :icon="Refresh"></el-button>
  235. </el-button-group>
  236. </div>
  237. <el-divider />
  238. <el-tree
  239. ref="tree$"
  240. :data="langDataSource"
  241. node-key="key"
  242. default-expand-all
  243. :expand-on-click-node="false"
  244. @node-click="onNodeClick"
  245. draggable
  246. :allow-drop="allowDrop"
  247. @node-drag-end="onDragEnd"
  248. >
  249. <template #default="{ node, data }">
  250. <Node
  251. :node="node"
  252. :data="data"
  253. @append="() => onAppend(data)"
  254. @remove="() => onRemove(node, data)"
  255. />
  256. </template>
  257. </el-tree>
  258. </div>
  259. <MdEditor
  260. v-loading="textLoading"
  261. :disabled="!currentOpenData"
  262. v-model="text"
  263. @save="onSave"
  264. @uploadImg="onUploadImg"
  265. />
  266. </div>
  267. <AppendModal
  268. v-model:show="showAppendModal"
  269. :contextData="appendContextData"
  270. :appendLoading="appendLoading"
  271. @ok="(formData) => resolveAppend?.(formData)"
  272. @cancel="() => rejectAppend?.()"
  273. />
  274. </template>
  275. <style lang="scss" scoped>
  276. .md-container {
  277. width: 100vw;
  278. height: 100dvh;
  279. display: flex;
  280. align-items: stretch;
  281. .md-editor {
  282. flex: 1;
  283. height: 100%;
  284. }
  285. .left {
  286. flex: 0 0 300px;
  287. padding: 10px;
  288. display: flex;
  289. flex-direction: column;
  290. align-items: stretch;
  291. .toolbar {
  292. display: flex;
  293. flex-direction: column;
  294. gap: 5px;
  295. }
  296. .el-tree {
  297. overflow: auto;
  298. flex: 1;
  299. :deep(.el-tree-node.is-current) {
  300. & > .el-tree-node__content {
  301. background-color: aqua;
  302. }
  303. }
  304. }
  305. }
  306. }
  307. </style>