123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426 |
- <template>
- <div class="header">
- <a-upload
- accept="image/*"
- multiple
- :beforeUpload="beforeUpload"
- :showUploadList="false"
- name="file"
- >
- <a-button
- v-if="butText"
- @click="addImage"
- :disabled="props.disabled"
- type="primary"
- size="samll"
- class="mr-10"
- >{{ butText }}</a-button
- >
- </a-upload>
- <div style="width: 300px" v-if="list.length">
- <a-slider
- :max="300"
- :min="60"
- id="test"
- v-model:value="baseSize"
- @change="changeZoom"
- />
- </div>
- </div>
- <div
- class="picture-wall-box"
- :style="{
- gridTemplateColumns: `repeat(auto-fill, ${baseSize}px)`,
- }"
- >
- <a-image-preview-group>
- <div
- class="picture-wall-item"
- v-for="(item, index) in list"
- :key="item.attachmentid"
- :style="{
- width: baseSize + 'px',
- height: baseSize + 'px',
- }"
- :draggable="!props.disabled && pic"
- @dragstart="dragStart(index)"
- @dragover="dragOver(index)"
- @dragend="dragEnd"
- >
- <div v-if="item.uid">
- <a-progress
- type="circle"
- :percent="item.progress"
- :status="item.exception"
- :size="[baseSize - 40, baseSize - 40]"
- />
- </div>
- <a-popover
- v-else
- :open="false && hovered == index"
- @openChange="handleHoverChange($event, index)"
- >
- <template #content>
- <a-image
- :src="item.cover"
- width="200px"
- :previewMask="false"
- fallback=""
- >
- <template #placeholder>
- <a-image :src="item.cover" :width="200" :preview="false" />
- </template>
- </a-image>
- </template>
- <div>
- <a-image
- :src="item.cover"
- :preview="{
- src: item.url,
- }"
- :style="{
- width: baseSize - (baseSize > 100 ? 30 : 15) + 'px',
- height: baseSize - (baseSize > 100 ? 30 : 15) + 'px',
- objectFit: 'cover',
- fontSize: 0,
- }"
- fallback=""
- >
- <template #previewMask>
- <a-popconfirm
- v-if="!props.disabled"
- :open="linksid == item.linksid"
- title="确定删除当前资源吗?"
- @confirm="confirmDeleteImage"
- @cancel="linksid = null"
- >
- <DeleteOutlined
- v-if="isDeletable"
- class="previewMaskIcon"
- :style="{ fontSize: (baseSize > 200 ? '24px' : '20px') }"
- @click.stop="linksid = item.linksid"
- />
- </a-popconfirm>
- </template>
- </a-image>
- </div>
- </a-popover>
- </div>
- </a-image-preview-group>
- </div>
- <div class="empty" v-if="!list.length">
- <a-empty :image="simpleImage" />
- </div>
- </template>
- <script setup>
- import { ref, reactive, defineProps, watch } from "vue";
- import Api from "@/api/api";
- import Up from "@/api/upload";
- import utils from "@/utils/utils";
- import { ExpandOutlined, DeleteOutlined } from "@ant-design/icons-vue";
- import { Empty } from "ant-design-vue";
- const simpleImage = Empty.PRESENTED_IMAGE_SIMPLE;
- const props = defineProps({
- butText: {
- type: String,
- default: "添加图片",
- },
- disabled: {
- type: Boolean,
- default: true,
- },
- attinfos: {
- type: Array,
- },
- coverType: {
- //列表显示的类型
- type: String,
- default: "thumbnail",
- },
- pic: {
- //pic支持排序
- type: Boolean,
- default: false,
- },
- startSequence: {
- //设置排序的起量,比如说第二页每页20条数据 起量为21
- type: [String, Number],
- default: 0,
- },
- total: {
- //考虑分页情况传递,上传图片顺序会在此向后顺延
- type: [Number, String],
- default: 0,
- },
- ownertable: {
- type: String,
- },
- ownerid: {
- type: [String, Number],
- },
- isDeletable: {
- type: Boolean,
- default: true,
- },
- });
- let baseSize = ref("140"),
- zoom = ref("50"),
- list = ref([]);
- const appfolderid = ref("");
- Api.requested({
- classname: "webmanage.site.site",
- method: "querySite_Parameter",
- content: {},
- }).then((res) => {
- appfolderid.value = res.data.appfolderid;
- });
- function beforeUpload(file) {
- file.document = file.name;
- const filetype = file.document.substr(file.document.lastIndexOf(".") + 1);
- file.sequence = list.value.length;
- file.progress = 0;
- file.linksid = -1;
- file.attachmentid = Date.now();
- list.value.push(reactive({ ...file }));
- Api.requested({
- classname: "system.attachment.huawei.OBS",
- method: "getFileName",
- content: {
- filename: file.name,
- filetype,
- parentid: appfolderid.value, //归属文件夹ID
- },
- }).then((res) => {
- if (res.code != 1)
- return (list.value[file.sequence].exception = "exception");
- console.log("获取上传链接", res);
- const { uploadurl, serialfilename } = res.data;
- Up.upload(uploadurl, file, {
- headers: {
- "Content-Type": "application/octet-stream",
- },
- onUploadProgress: function (e) {
- list.value[file.sequence].progress = parseInt(e.progress * 100);
- console.log(list.value[file.sequence]);
- },
- }).then((upload) => {
- console.log("上传结果", upload);
- if (upload.status != 200)
- return (list.value[file.sequence].exception = "exception");
- Api.requested({
- classname: "system.attachment.huawei.OBS",
- method: "uploadSuccess",
- content: {
- serialfilename,
- ownerid: props.ownerid,
- ownertable: props.ownertable,
- usetype: "file",
- },
- }).then((feedback) => {
- console.log("feedback", feedback);
- if (feedback.code != 1)
- return (list.value[file.sequence].exception = "exception");
- const attachmentid = feedback.data.attachmentids[0];
- Api.requested({
- id: 20240407135802,
- content: {
- ownerid: props.ownerid,
- ownertable: props.ownertable,
- attachmentid,
- sequence: file.sequence,
- },
- }).then((binding) => {
- console.log(binding, "binding");
- if (binding.code != 1)
- return (list.value[file.sequence].exception = "exception");
- Api.requested({
- id: "20240407140002",
- content: {
- ownerid: props.ownerid,
- ownertable: props.ownertable,
- pageNumber: 1,
- pageSize: 100,
- where: {
- // condition: serialfilename,
- },
- },
- }).then((getList) => {
- if (getList.code != 1)
- return (list.value[file.sequence].exception = "exception");
- list.value[file.sequence] = getList.data
- .filter((v) => v.attachmentid == attachmentid)
- .map((v) => {
- v = Object.assign(v, v.attinfos[0]);
- delete v.attinfos;
- v.cover = props.coverType
- ? v.subfiles.find((s) => s.type == props.coverType).url ||
- v.url
- : v.url;
- return v;
- })[0];
- });
- });
- });
- });
- });
- }
- /* 初始化列表 */
- watch(
- () => props.attinfos,
- (newVal) => {
- init(newVal);
- },
- () => showStyle,
- (newVal) => {
- console.log("showStyle", newVal);
- }
- );
- function init(attinfos, pic = props.pic) {
- if(!attinfos.length) return;
- list.value = pic ? attinfos : attinfos.filter((v) => v.fileType == "image");
- list.value = list.value.map((v) => {
- if (pic) {
- v = Object.assign(v, v.attinfos[0]);
- delete v.attinfos;
- }
- v.cover = props.coverType
- ? v.subfiles.find((s) => s.type == props.coverType).url || v.url
- : v.url;
- return v;
- });
- if (
- pic &&
- list.value[list.value.length - 1].sequence !=
- props.startSequence + list.value.length
- ) {
- setSequenceAll();
- }
- }
- function addImage() {
- console.log("添加图片");
- }
- let linksid = ref(null);
- async function confirmDeleteImage() {
- await handleDeleteImage([linksid.value]);
- linksid.value = null;
- }
- /* 处理删除图片 */
- function handleDeleteImage(linksids = [], setSequence = true) {
- return new Promise((resolve, reject) => {
- if (linksids.length) {
- Api.requested({
- id: "20240407135902",
- content: { linksids },
- }).then((res) => {
- console.log("删除图片", res);
- utils.message(res, "操作成功", async () => {
- list.value = list.value.filter((v) => !linksids.includes(v.linksid));
- //如果考虑分页情况再这个节点去重新获取数据 setSequence=false
- if (setSequence) await setSequenceAll();
- });
- resolve(res.code == 1);
- });
- } else {
- resolve();
- }
- });
- }
- /* 拖拽与显示气泡 */
- let startIndex = null,
- targetIndex = null,
- hovered = ref(null),
- prohibitHovered = false;
- function dragStart(index) {
- startIndex = index;
- targetIndex = index;
- prohibitHovered = true;
- hovered.value = null;
- }
- function dragOver(index) {
- if (targetIndex == index) return;
- let item = list.value.splice(targetIndex, 1);
- list.value.splice(index, 0, item[0]);
- targetIndex = index;
- }
- function dragEnd() {
- prohibitHovered = false;
- if (startIndex != targetIndex) setSequenceAll();
- }
- function setSequenceAll() {
- return new Promise((resolve, reject) => {
- Api.requested({
- id: "20221201134901",
- content: {
- ownertable: "sys_attachment_links",
- sequencesorts: list.value.map((v, i) => {
- return {
- ownerid: v.linksid,
- sequence: Number(props.startSequence) + i,
- };
- }),
- },
- }).then((res) => {
- console.log("设置排序", res);
- resolve(res.code == 1);
- });
- });
- }
- function handleHoverChange(visible, index) {
- if (prohibitHovered) return (hovered.value = null);
- hovered.value = visible ? index : null;
- }
- </script>
- <style scoped>
- .header {
- display: flex;
- justify-content: space-between;
- align-items: center;
- width: 100%;
- }
- .picture-wall-box {
- display: grid;
- justify-content: space-between;
- grid-gap: 10px;
- margin-top: 20px;
- user-select: none;
- }
- .picture-wall-box .picture-wall-item {
- display: flex;
- align-items: center;
- justify-content: center;
- border: 1px solid #d9d9d9;
- border-radius: 8px;
- box-sizing: border-box;
- }
- .previewMaskIcon {
- opacity: 0.6;
- }
- .previewMaskIcon:hover {
- opacity: 1;
- }
- .empty {
- width: 100%;
- padding: 50px 0;
- display: flex;
- justify-content: center;
- }
- </style>
|