index.vue 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583
  1. <template>
  2. <div class="material" v-loading="pager.loading">
  3. <div class="material__left">
  4. <div class="flex-1 min-h-0">
  5. <el-scrollbar>
  6. <div class="material-left__content pt-4 p-b-4">
  7. <el-tree
  8. ref="treeRef"
  9. node-key="id"
  10. :data="cateLists"
  11. empty-text="''"
  12. :highlight-current="true"
  13. :expand-on-click-node="false"
  14. :current-node-key="cateId"
  15. @node-click="handleCatSelect"
  16. >
  17. <template v-slot="{ data }">
  18. <div class="flex flex-1 items-center min-w-0 pr-4">
  19. <img
  20. class="w-[20px] h-[16px] mr-3"
  21. src="@/assets/images/icon_folder.png"
  22. />
  23. <span class="flex-1 truncate mr-2">
  24. <overflow-tooltip :content="data.name" />
  25. </span>
  26. <el-dropdown
  27. v-perms="['albums:cateRename', 'albums:cateDel']"
  28. v-if="data.id > 0"
  29. :hide-on-click="false"
  30. >
  31. <span class="muted m-r-10">···</span>
  32. <template #dropdown>
  33. <el-dropdown-menu>
  34. <popover-input
  35. v-perms="['albums:cateRename']"
  36. @confirm="handleEditCate($event, data.id)"
  37. size="default"
  38. :value="data.name"
  39. width="400px"
  40. :limit="20"
  41. show-limit
  42. teleported
  43. >
  44. <div>
  45. <el-dropdown-item>
  46. 命名分组
  47. </el-dropdown-item>
  48. </div>
  49. </popover-input>
  50. <div
  51. v-perms="['albums:cateDel']"
  52. @click="handleDeleteCate(data.id)"
  53. >
  54. <el-dropdown-item>删除分组</el-dropdown-item>
  55. </div>
  56. </el-dropdown-menu>
  57. </template>
  58. </el-dropdown>
  59. </div>
  60. </template>
  61. </el-tree>
  62. </div>
  63. </el-scrollbar>
  64. </div>
  65. <div class="flex justify-center p-2 border-t border-br">
  66. <popover-input
  67. v-perms="['albums:cateAdd']"
  68. @confirm="handleAddCate"
  69. size="default"
  70. width="400px"
  71. :limit="20"
  72. show-limit
  73. teleported
  74. >
  75. <el-button> 添加分组 </el-button>
  76. </popover-input>
  77. </div>
  78. </div>
  79. <div class="material__center flex flex-col">
  80. <div class="operate-btn flex">
  81. <div class="flex-1 flex">
  82. <upload
  83. v-if="type == 'image'"
  84. v-perms="['upload:image']"
  85. class="mr-3"
  86. :data="{ cid: cateId,brandId:brandId,houseId:houseId }"
  87. :type="type"
  88. :show-progress="true"
  89. @change="refresh"
  90. >
  91. <el-button type="primary">本地上传</el-button>
  92. </upload>
  93. <upload
  94. v-if="type == 'video'"
  95. v-perms="['upload:video']"
  96. class="mr-3"
  97. :data="{ cid: cateId,brandId:brandId,houseId:houseId }"
  98. :type="type"
  99. :show-progress="true"
  100. @change="refresh"
  101. >
  102. <el-button type="primary">本地上传</el-button>
  103. </upload>
  104. <el-button
  105. v-perms="['albums:albumDel']"
  106. v-if="mode == 'page'"
  107. :disabled="!select.length"
  108. @click.stop="batchFileDelete()"
  109. >
  110. 删除
  111. </el-button>
  112. <popup
  113. v-perms="['albums:albumMove']"
  114. v-if="mode == 'page'"
  115. class="ml-3"
  116. @confirm="batchFileMove"
  117. :disabled="!select.length"
  118. title="移动文件"
  119. >
  120. <template #trigger>
  121. <el-button :disabled="!select.length">移动</el-button>
  122. </template>
  123. <div>
  124. <span class="mr-5">移动文件至</span>
  125. <el-select v-model="moveId" placeholder="请选择">
  126. <template v-for="item in cateLists" :key="item.id">
  127. <el-option
  128. v-if="item.id !== ''"
  129. :label="item.name"
  130. :value="item.id"
  131. ></el-option>
  132. </template>
  133. </el-select>
  134. </div>
  135. </popup>
  136. </div>
  137. <el-input
  138. class="w-60"
  139. placeholder="请输入名称"
  140. v-model="fileParams.name"
  141. @keyup.enter="refresh"
  142. >
  143. <template #append>
  144. <el-button @click="refresh">
  145. <template #icon>
  146. <icon name="el-icon-Search" />
  147. </template>
  148. </el-button>
  149. </template>
  150. </el-input>
  151. <div class="flex items-center ml-2">
  152. <el-tooltip content="列表视图" placement="top">
  153. <div
  154. class="list-icon"
  155. :class="{
  156. select: listShowType == 'table'
  157. }"
  158. @click="listShowType = 'table'"
  159. >
  160. <icon name="local-icon-list-2" :size="18" />
  161. </div>
  162. </el-tooltip>
  163. <el-tooltip content="平铺视图" placement="top">
  164. <div
  165. class="list-icon"
  166. :class="{
  167. select: listShowType == 'normal'
  168. }"
  169. @click="listShowType = 'normal'"
  170. >
  171. <icon name="el-icon-Menu" :size="18" />
  172. </div>
  173. </el-tooltip>
  174. </div>
  175. </div>
  176. <div class="mt-3" v-if="mode == 'page'">
  177. <el-checkbox
  178. :disabled="!pager.lists.length"
  179. v-model="isCheckAll"
  180. @change="selectAll"
  181. :indeterminate="isIndeterminate"
  182. >
  183. 当页全选
  184. </el-checkbox>
  185. </div>
  186. <div class="material-center__content flex flex-col flex-1 mb-1 min-h-0">
  187. <el-scrollbar v-if="pager.lists.length" v-show="listShowType == 'normal'">
  188. <ul class="file-list flex flex-wrap mt-4">
  189. <li
  190. class="file-item-wrap"
  191. v-for="item in pager.lists"
  192. :key="item.id"
  193. :style="{ width: fileSize }"
  194. >
  195. <del-wrap @close="batchFileDelete([item.id])">
  196. <file-item
  197. :uri="item.uri"
  198. :file-size="fileSize"
  199. :type="type"
  200. @click="selectFile(item)"
  201. >
  202. <div class="item-selected" v-if="isSelect(item.id)">
  203. <icon :size="24" name="el-icon-Check" color="#fff" />
  204. </div>
  205. </file-item>
  206. </del-wrap>
  207. <overflow-tooltip class="mt-1" :content="item.name" />
  208. <div class="operation-btns flex items-center">
  209. <popover-input
  210. v-perms="['albums:albumRename']"
  211. @confirm="handleFileRename($event, item.id)"
  212. size="default"
  213. :value="item.name"
  214. width="400px"
  215. :limit="50"
  216. show-limit
  217. teleported
  218. >
  219. <el-button type="primary" link> 重命名 </el-button>
  220. </popover-input>
  221. <el-button type="primary" link @click="handlePreview(item.uri)">
  222. 查看
  223. </el-button>
  224. </div>
  225. </li>
  226. </ul>
  227. </el-scrollbar>
  228. <el-table
  229. ref="tableRef"
  230. class="mt-4"
  231. v-show="listShowType == 'table'"
  232. :data="pager.lists"
  233. width="100%"
  234. height="100%"
  235. size="large"
  236. @row-click="selectFile"
  237. >
  238. <el-table-column width="55">
  239. <template #default="{ row }">
  240. <el-checkbox :modelValue="isSelect(row.id)" @change="selectFile(row)" />
  241. </template>
  242. </el-table-column>
  243. <el-table-column label="图片" width="100">
  244. <template #default="{ row }">
  245. <file-item :uri="row.uri" file-size="50px" :type="type"></file-item>
  246. </template>
  247. </el-table-column>
  248. <el-table-column label="名称" min-width="100" show-overflow-tooltip>
  249. <template #default="{ row }">
  250. <el-link @click.stop="handlePreview(row.uri)" :underline="false">
  251. {{ row.name }}
  252. </el-link>
  253. </template>
  254. </el-table-column>
  255. <el-table-column prop="createTime" label="上传时间" min-width="100" />
  256. <el-table-column label="操作" width="150" fixed="right">
  257. <template #default="{ row }">
  258. <div class="inline-block" v-perms="['albums:albumRename']">
  259. <popover-input
  260. @confirm="handleFileRename($event, row.id)"
  261. size="default"
  262. :value="row.name"
  263. width="400px"
  264. :limit="50"
  265. show-limit
  266. teleported
  267. >
  268. <el-button type="primary" link> 重命名 </el-button>
  269. </popover-input>
  270. </div>
  271. <div class="inline-block">
  272. <el-button type="primary" link @click.stop="handlePreview(row.uri)">
  273. 查看
  274. </el-button>
  275. </div>
  276. <div class="inline-block" v-perms="['albums:albumDel']">
  277. <el-button
  278. type="primary"
  279. link
  280. @click.stop="batchFileDelete([row.id])"
  281. >
  282. 删除
  283. </el-button>
  284. </div>
  285. </template>
  286. </el-table-column>
  287. </el-table>
  288. <div
  289. class="flex flex-1 justify-center items-center"
  290. v-if="!pager.loading && !pager.lists.length"
  291. >
  292. 暂无数据~
  293. </div>
  294. </div>
  295. <div class="material-center__footer flex justify-between items-center mt-2">
  296. <div class="flex">
  297. <template v-if="mode == 'page'">
  298. <span class="mr-3">
  299. <el-checkbox
  300. :disabled="!pager.lists.length"
  301. v-model="isCheckAll"
  302. @change="selectAll"
  303. :indeterminate="isIndeterminate"
  304. >
  305. 当页全选
  306. </el-checkbox>
  307. </span>
  308. <el-button
  309. v-perms="['albums:albumDel']"
  310. :disabled="!select.length"
  311. @click="batchFileDelete()"
  312. >
  313. 删除
  314. </el-button>
  315. <popup
  316. v-perms="['albums:albumMove']"
  317. class="ml-3 inline"
  318. @confirm="batchFileMove"
  319. :disabled="!select.length"
  320. title="移动文件"
  321. >
  322. <template #trigger>
  323. <el-button :disabled="!select.length">移动</el-button>
  324. </template>
  325. <div>
  326. <span class="mr-5">移动文件至</span>
  327. <el-select v-model="moveId" placeholder="请选择">
  328. <template v-for="item in cateLists" :key="item.id">
  329. <el-option
  330. v-if="item.id !== ''"
  331. :label="item.name"
  332. :value="item.id"
  333. ></el-option>
  334. </template>
  335. </el-select>
  336. </div>
  337. </popup>
  338. </template>
  339. </div>
  340. <pagination
  341. v-model="pager"
  342. @change="getFileList"
  343. layout="total, prev, pager, next, jumper"
  344. />
  345. </div>
  346. </div>
  347. <div class="material__right" v-if="mode == 'picker'">
  348. <div class="flex justify-between p-2 flex-wrap">
  349. <div class="sm flex items-center">
  350. 已选择 {{ select.length }}
  351. <span v-if="limit">/{{ limit }}</span>
  352. </div>
  353. <el-button type="primary" link @click="clearSelect">清空</el-button>
  354. </div>
  355. <div class="flex-1 min-h-0">
  356. <el-scrollbar class="ls-scrollbar">
  357. <ul class="select-lists flex flex-col p-t-3">
  358. <li class="mb-4" v-for="item in select" :key="item.id">
  359. <div class="select-item">
  360. <del-wrap @close="cancelSelete(item.id)">
  361. <file-item
  362. :uri="item.uri"
  363. file-size="100px"
  364. :type="type"
  365. ></file-item>
  366. </del-wrap>
  367. </div>
  368. </li>
  369. </ul>
  370. </el-scrollbar>
  371. </div>
  372. </div>
  373. <preview v-model="showPreview" :url="previewUrl" :type="type" />
  374. </div>
  375. </template>
  376. <script lang="ts" setup>
  377. import { useCate, useFile } from './hook'
  378. import FileItem from './file.vue'
  379. import Preview from './preview.vue'
  380. import type { Ref } from 'vue'
  381. import {getBrandId,getHouseId } from '@/utils/auth'
  382. const props = defineProps({
  383. fileSize: {
  384. type: String,
  385. default: '100px'
  386. },
  387. limit: {
  388. type: Number,
  389. default: 1
  390. },
  391. type: {
  392. type: String,
  393. default: 'image'
  394. },
  395. mode: {
  396. type: String,
  397. default: 'picker'
  398. },
  399. pageSize: {
  400. type: Number,
  401. default: 15
  402. }
  403. })
  404. const brandId = ref(getBrandId())
  405. const houseId = ref(getHouseId())
  406. const emit = defineEmits(['change'])
  407. const { limit } = toRefs(props)
  408. const typeValue = computed<number>(() => {
  409. switch (props.type) {
  410. case 'image':
  411. return 10
  412. case 'video':
  413. return 20
  414. case 'file':
  415. return 30
  416. default:
  417. return 0
  418. }
  419. })
  420. const visible: Ref<boolean> = inject('visible')!
  421. const previewUrl = ref('')
  422. const showPreview = ref(false)
  423. const {
  424. treeRef,
  425. cateId,
  426. cateLists,
  427. handleAddCate,
  428. handleEditCate,
  429. handleDeleteCate,
  430. getCateLists,
  431. handleCatSelect
  432. } = useCate(typeValue.value)
  433. const {
  434. tableRef,
  435. listShowType,
  436. moveId,
  437. pager,
  438. fileParams,
  439. select,
  440. isCheckAll,
  441. isIndeterminate,
  442. getFileList,
  443. refresh,
  444. batchFileDelete,
  445. batchFileMove,
  446. selectFile,
  447. isSelect,
  448. clearSelect,
  449. cancelSelete,
  450. selectAll,
  451. handleFileRename
  452. } = useFile(cateId, typeValue, limit, props.pageSize)
  453. const getData = async () => {
  454. await getCateLists()
  455. treeRef.value?.setCurrentKey(cateId.value)
  456. getFileList()
  457. }
  458. const handlePreview = (url: string) => {
  459. previewUrl.value = url
  460. showPreview.value = true
  461. }
  462. watch(
  463. visible,
  464. async (val: boolean) => {
  465. if (val) {
  466. getData()
  467. }
  468. },
  469. {
  470. immediate: true
  471. }
  472. )
  473. watch(cateId, () => {
  474. fileParams.name = ''
  475. refresh()
  476. })
  477. watch(
  478. select,
  479. (val: any[]) => {
  480. emit('change', val)
  481. if (val.length == pager.lists.length && val.length !== 0) {
  482. isIndeterminate.value = false
  483. isCheckAll.value = true
  484. return
  485. }
  486. if (val.length > 0) {
  487. isIndeterminate.value = true
  488. } else {
  489. isCheckAll.value = false
  490. isIndeterminate.value = false
  491. }
  492. },
  493. {
  494. deep: true
  495. }
  496. )
  497. onMounted(() => {
  498. props.mode == 'page' && getData()
  499. })
  500. defineExpose({
  501. clearSelect
  502. })
  503. </script>
  504. <style scoped lang="scss">
  505. .material {
  506. @apply h-full min-h-0 flex flex-1;
  507. &__left {
  508. @apply border-r border-br flex flex-col w-[200px];
  509. :deep(.el-tree-node__content) {
  510. height: 36px;
  511. }
  512. }
  513. &__center {
  514. flex: 1;
  515. min-width: 0;
  516. min-height: 0;
  517. padding: 16px 16px 0;
  518. .list-icon {
  519. border-radius: 3px;
  520. display: flex;
  521. padding: 5px;
  522. cursor: pointer;
  523. &.select {
  524. @apply text-primary bg-primary-light-8;
  525. }
  526. }
  527. .file-list {
  528. .file-item-wrap {
  529. margin-right: 16px;
  530. line-height: 1.3;
  531. cursor: pointer;
  532. .item-selected {
  533. display: flex;
  534. align-items: center;
  535. justify-content: center;
  536. position: absolute;
  537. top: 0;
  538. left: 0;
  539. width: 100%;
  540. height: 100%;
  541. border-radius: 4px;
  542. background-color: rgba(0, 0, 0, 0.5);
  543. box-sizing: border-box;
  544. }
  545. .operation-btns {
  546. height: 28px;
  547. visibility: hidden;
  548. }
  549. &:hover .operation-btns {
  550. visibility: visible;
  551. }
  552. }
  553. }
  554. }
  555. &__right {
  556. @apply border-l border-br flex flex-col;
  557. width: 130px;
  558. .select-lists {
  559. padding: 10px;
  560. .select-item {
  561. width: 100px;
  562. height: 100px;
  563. }
  564. }
  565. }
  566. }
  567. </style>