国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

web前端之拖拽API、上傳多圖片時拖拽排序、表格行或列拖拽排序、復制元素跨區(qū)域放置、拖放、投擲、若依、vuedraggable、sortablejs、element、plus、vue、ui

這篇具有很好參考價值的文章主要介紹了web前端之拖拽API、上傳多圖片時拖拽排序、表格行或列拖拽排序、復制元素跨區(qū)域放置、拖放、投擲、若依、vuedraggable、sortablejs、element、plus、vue、ui。希望對大家有所幫助。如果存在錯誤或未考慮完全的地方,請大家不吝賜教,您也可以點擊"舉報違法"按鈕提交疑問。


vue3+element-plus+table+sortablejs實現(xiàn)行或列的拖拽

前言

vue3+element-puls列表行、列拖拽的需求,想找一個成熟的解決方法。但發(fā)現(xiàn)vue3的比較少,所以就把這個分享出來,希望可以幫助到大家。vuedraggable是一款vue3的拖拽插件,基于sortable.js實現(xiàn),可以用來拖拽列表、菜單、工作臺、選項卡等常見的工作場景。安裝的是vuedraggable@next,引入使用的是sortable.js。

npm install vuedraggable@next -S
yarn add vuedraggable@next

html

<el-row :gutter="20">
  <el-col :span="24">行拖動排序</el-col>
  <el-col class="mt_10" :span="24">
    <el-table id="idRow" :data="tableList1" :row-key="(row) => row.id" border>
      <el-table-column label="姓名" prop="name" />
      <el-table-column label="性別" width="100" align="center">
        <template #default="scope">
          <span>{{ scope.row.sex === 0 ? "女" : "男" }}</span>
        </template>
      </el-table-column>
      <el-table-column label="年齡" prop="age" width="100" align="center" />
    </el-table>
  </el-col>
  <el-col class="mt_68" :span="24">列拖動排序</el-col>
  <el-col class="mt_10" :span="24">
    <div class="draggable-table">
      <el-table id="idColumn" ref="refColumn" :data="tableList2" :key="keyTable" border>
        <template v-for="item in tableIColumn" :key="item.id">
          <el-table-column :label="item.label" :prop="item.prop" :width="item.width" :align="item.align" />
        </template>
      </el-table>
    </div>
  </el-col>
</el-row>

表格標簽一定要加上唯一值。如果沒有,則數(shù)據(jù)更新,但界面沒有效果。
行的唯一值 <=> :row-key="(row) => row.id"
列的唯一值 <=> :key="keyTable"
當需要在一個表內(nèi)同時實現(xiàn)行或列拖拽時,需要把行和列的唯一值都寫在el-table上才起作用。


JavaScript

import Sortable from "sortablejs";

let tableList1 = ref([
  { id: "a1", name: "何二", sex: 0, age: 18 },
  { id: "b2", name: "張三", sex: 1, age: 20 },
  { id: "c3", name: "李四", sex: 1, age: 20 },
  { id: "d4", name: "王五", sex: 0, age: 25 },
  { id: "e5", name: "趙六", sex: 1, age: 56 },
  { id: "f6", name: "田七", sex: 1, age: 35 },
]),
  tableList2 = ref([
    { id: "a1", name: "何二", sex: 0, age: 18 },
    { id: "b2", name: "張三", sex: 1, age: 20 },
    { id: "d4", name: "王五", sex: 0, age: 25 },
    { id: "e5", name: "趙六", sex: 1, age: 56 },
  ]);

// 拖動列排序
let refColumn = ref(),
  keyTable = ref(),
  tableIColumn = ref([
    { label: "姓名", prop: "name", width: "", align: "" },
    { label: "性別", prop: "sex", width: "68", align: "center" },
    { label: "年齡", prop: "age", width: "68", align: "center" },
  ]);

/**
 * 列拖動排序初始化
 */
function handleInitColumn() {
  nextTick(() => {
    let refEl = refColumn.value.$el.querySelector(".el-table__header-wrapper tr");

    Sortable.create(refEl, {
      animation: 150,
      onEnd(event) {
        const oldItem = tableIColumn.value[event.oldIndex];

        tableIColumn.value.splice(event.oldIndex, 1);
        tableIColumn.value.splice(event.newIndex, 0, oldItem);

        keyTable.value = "key" + new Date().getTime();
        nextTick(() => {
          // 因為table被強制重新繪制
          // 因此需要重新監(jiān)聽
          handleInitColumn();
        });
      },
    });
  });
}

/**
 * 行拖動排序初始化
 */
function handleInitRow() {
  // 要拖拽元素的父容器
  let tbody = document.querySelector("#idRow .el-table__body-wrapper tbody");

  new Sortable(tbody, {
    // 可被拖拽的子元素
    draggable: "#idRow .el-table__row",
    /**
     * 開始移動
     * @param {Object} event
     */
    onStart: function (event) {
      event.item.classList.add("b_lg01_i");
    },
    /**
     * 移動過程
     * @param {Object} event
     */
    // onMove(event) {
    //   // 設(shè)置經(jīng)過的行的背景色為灰色(此方式不起作用,因為樣式權(quán)重問題)
    //   event.related.classList.add("b_lg01_i");
    // },
    /**
     * 移動結(jié)束(放置)
     * @param {Object} event
     */
    onEnd({ item, newIndex, oldIndex }) {
      // 設(shè)置經(jīng)過的行的背景色為灰色(此方式不起作用,因為樣式權(quán)重問題)
      // item.style.backgroundColor = 'initial';

      const currRow = tableList1.value.splice(oldIndex, 1)[0];

      tableList1.value.splice(newIndex, 0, currRow);
      item.classList.remove("b_lg01_i");
    },
  });
}

nextTick(() => {
  handleInitColumn();
  handleInitRow();
});

代碼中的每一個nextTick都有它的用處,一個都不能少,否則有bug。


style

.b_lg01_i {
  background: linear-gradient(to top, #fffaf0, #ffdead) !important;
}

.bc_initial_i {
  background-color: initial !important;
}

添加lang="scss" scoped屬性需謹慎,這會導致樣式不起作用。


web前端之實現(xiàn)拖拽放置、復制元素

效果圖

vue 下載snail插件 命令,web前端,vue3,element-plus,前端,vue.js,web,element,vuedraggable


vue 下載snail插件 命令,web前端,vue3,element-plus,前端,vue.js,web,element,vuedraggable


html

<div class="container">
    <div class="left" data-drop="move">
        <div data-effect="copy" draggable="true" class="item color1">語文</div>
        <div data-effect="copy" draggable="true" class="item color2">數(shù)學</div>
        <div data-effect="copy" draggable="true" class="item color3">英語</div>
        <div data-effect="copy" draggable="true" class="item color4">音樂</div>
        <div data-effect="copy" draggable="true" class="item color5">政治</div>
        <div data-effect="copy" draggable="true" class="item color6">歷史</div>
        <div data-effect="copy" draggable="true" class="item color7">體育</div>
    </div>

    <div class="right">
        <table border>
            <colgroup>
                <col />
                <col />
                <col />
                <col />
                <col />
                <col />
                <col />
                <col />
                <col />
            </colgroup>
            <thead>
                <tr>
                    <td>課表</td>
                    <th>星期一</th>
                    <th>星期二</th>
                    <th>星期三</th>
                    <th>星期四</th>
                    <th>星期五</th>
                    <th>星期六</th>
                    <th>星期天</th>
                </tr>
            </thead>
            <tbody>
                <tr>
                    <th rowspan="3" class="span">上午</th>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                </tr>
                <tr>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                </tr>
                <tr>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                </tr>
                <tr>
                    <th rowspan="4" class="span">下午</th>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                </tr>
                <tr>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                </tr>
                <tr>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                </tr>
                <tr>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                    <td data-drop="copy"></td>
                </tr>
            </tbody>
        </table>
    </div>
</div>

JavaScrip

const container = document.querySelector('.container');
// 移動的元素
let source = undefined;

/**
 * 移動開始
 * @param {Element} e 元素
 */
container.ondragstart = (e) => {
    // 設(shè)置鼠標樣式
    e.dataTransfer.effectAllowed = e.target.dataset.effect;
    source = e.target;
}

/**
 * 移動過程
 * @param {Element} e 元素
 */
container.ondragover = (e) => {
    // 大部分標簽不允許把元素放到它上面
    // 導致ondrop事件不被觸發(fā)
    // 這個是阻止標簽?zāi)J行為
    e.preventDefault();
}

/**
 * 當前子元素如果不存在該屬性,繼續(xù)尋找父元素,直到找到為止
 * @param {*} node 元素節(jié)點
 * @returns 帶匹配的自定義屬性的元素節(jié)點
 */
function getDropNode(node) {
    while (node) {
        if (node.dataset?.drop) return node;

        node = node.parentNode;
    }
}

/**
 * 清空背景
 */
function clearDropStyle() {
    const dropNodes = document.querySelectorAll('.drop_over');

    dropNodes.forEach((node) => {
        node.classList.remove('drop_over');
    });
}

/**
 * 移動結(jié)束
 * @param {Element} e 元素
 * @returns null
 */
container.ondragenter = (e) => {
    clearDropStyle();

    const dropNode = getDropNode(e.target);

    if (!dropNode) return false;
    if (e.dataTransfer.effectAllowed === dropNode.dataset.drop) dropNode.classList.add('drop_over');
}

/**
 * 放置時觸發(fā)
 * @param {Element} e 元素
 * @returns null
 */
container.ondrop = (e) => {
    clearDropStyle();

    const dropNode = getDropNode(e.target);

    if (!dropNode) return false;
    if (e.dataTransfer.effectAllowed !== dropNode.dataset.drop) return false;
    if (dropNode.dataset.drop === 'copy') {
        dropNode.innerHTML = '';

        const cloned = source.cloneNode(true);

        cloned.dataset.effect = 'move';
        dropNode.appendChild(cloned);
    } else {
        source.remove();
    }
}

style

* {
    margin: 0;
    padding: 0;
    box-sizing: border-box;
}

body {
    height: 100vh;
    display: flex;
    flex-direction: column;
    align-items: center;
    justify-content: center;
}

.container {
    height: 268px;
    display: flex;
}

.left {
    height: 100%;
    margin-right: 5px;
    display: flex;
    flex-flow: column;
    justify-content: space-between;
    background-color: #efefef;
    padding: 8px;
}

.item {
    width: 40px;
    height: 24px;
    font-size: 16px;
    line-height: 24px;
    text-align: center;
    margin-left: auto;
    margin-right: auto;
}

.color1 {
    background-color: #e44387;
}

.color2 {
    background-color: #69f9a6;
}

.color3 {
    background-color: #e25d4b;
}

.color4 {
    background-color: #ffea60;
}

.color5 {
    background-color: #7ae5f6;
}

.color6 {
    background-color: #c55fef;
}

.color7 {
    background-color: #66fea7;
}

.right {
    height: 100%;
    margin-left: 5px;
    background-color: #efefef;
    padding: 8px;
    display: flex;
    align-items: center;
}

th {
    background-color: #afafaf;
}

td {
    width: 50px;
    height: 30px;
    text-align: center;
}

.drop_over {
    background-color: #cfcfcf;
}

vue2+html5+原生dom+原生JavaScript實現(xiàn)跨區(qū)域拖放

關(guān)鍵代碼

// 放
function drop(ev) {
	let data = ev.dataTransfer.getData("Text"),
		i = ev.path[1].getAttribute("i"),
		text = document.getElementById(data).cloneNode(true).innerText.trim();

	if (i == null) return alert('請放置在文件名上');
	if (app.fileS[i].divs.includes(text)) return alert('不能放重復數(shù)據(jù)');
	app.fileS[i].divs.push(text);
	for (let is = 0; is < app.fileS.length; is++) {
		if (i == is) {
			app.fileS[is].isShow = true;
		} else {
			app.fileS[is].isShow = false;
		}
	}
}

完整代碼

gitee(碼云) - mj01分支 - copyDragAndDrop 文件


vue+element實現(xiàn)跨區(qū)域復制拖放

html部分

<div id="app">
	<div style="margin-left: 60px;">
        <!-- 放置的數(shù)據(jù) -->
        <div>
            <!-- 自定義標簽屬性 :i=`${i}` ondrop 函數(shù)獲取的是對應(yīng)的放置標簽屬性 -->
            <div style="margin: 20px 0; padding: 12px 0;" v-for="(item, i) in fileS" :key="item.id" :i=`${i}`
                ondrop="drop(event)" ondragover="allowDrop(event)">
                <div style="font-weight: 700; cursor: pointer;" :i=`${i}` @click="openOff(i)">
                    <span :i=`${i}`>{{item.title}}</span>
                    <span style="cursor: pointer;" :i=`${i}` v-show="item.isShow"></span>
                    <span style="cursor: pointer;" :i=`${i}` v-show="!item.isShow"></span>
                </div>

                <div :i=`${i}` v-if="item.isShow">
                    <p :i=`${i}` style="padding: 6px 0; color: #777;" v-for="items in item.divs" :key="items">
                        <span :i=`${i}`>{{items}}</span>
                        <span :i=`${i}` style="padding-left: 10px; cursor: pointer;">×</span>
                    </p>
                </div>
            </div>
        </div>

        <!-- 源數(shù)據(jù) -->
        <div style="margin-top: 20px; border-top: 1px solid #333;">
            <div style="margin-top: 10px; cursor: pointer; color: #000;" v-for="item in 7" :id=`drag${item}`
                draggable="true" ondragstart="drag(event)">
                源數(shù)據(jù) {{item}}</div>
        </div>
    </div>
</div>

JavaScript部分

let app = new Vue({
    el: "#app",
    data() {
        return {
            fileS: [
                { id: 1, title: "文件夾1", divs: [], isShow: false },
                { id: 2, title: "文件夾2", divs: [], isShow: false },
                { id: 3, title: "文件夾3", divs: [], isShow: false }
            ]
        }
    },

    methods: {
        openOff(i) {
            for (let is = 0; is < this.fileS.length; is++) {
                if (i == is && !this.fileS[is].isShow) {
                    this.fileS[is].isShow = true;
                } else {
                    this.fileS[is].isShow = false;
                }
            }
        }
    }
});

// 移動
function allowDrop(ev) {
    ev.preventDefault();
}

// 拖
function drag(ev) {
    ev.dataTransfer.setData("Text", ev.target.id);
}

// 放
function drop(ev) {
    let data = ev.dataTransfer.getData("Text"),
        i = ev.path[1].getAttribute("i"),
        text = document.getElementById(data).cloneNode(true).innerText.trim();

    if (i == null) return alert('請放置在文件名上');
    if (app.fileS[i].divs.includes(text)) return alert('不能放重復數(shù)據(jù)');
    app.fileS[i].divs.push(text);
    for (let is = 0; is < app.fileS.length; is++) {
    	app.fileS[is].isShow = i === is ? true : false;
    }
}

vue2實現(xiàn)跨區(qū)域拖放

關(guān)鍵代碼

dragend(item) {
	console.log(item);
	if (this.oldItem != this.newItem) {
		let oldIndex = this.List.indexOf(this.oldItem);
		let newIndex = this.List.indexOf(this.newItem);

		let oldflag = false
		let newflag = false

		if (oldIndex === -1) {
			oldflag = true
			oldIndex = this.list.indexOf(this.oldItem);
		}

		if (newIndex === -1) {
			newflag = true
			newIndex = this.list.indexOf(this.newItem);
		}

		let newList = [...this.List]; // 中間數(shù)組,用于交換兩個節(jié)點
		let newlist = [...this.list]; // 中間數(shù)組,用于交換兩個節(jié)點

		if (!oldflag) {
			newList.splice(oldIndex, 1);
		} else {
			newlist.splice(oldIndex, 1);
		}

		if (!newflag) {
			newList.splice(newIndex, 0, this.oldItem);
		} else {
			newlist.splice(newIndex, 0, this.oldItem);
		}

		// 刪除老的節(jié)點
		// newList.splice(oldIndex, 1);
		// // 在列表目標位置增加新的節(jié)點
		// newList.splice(newIndex, 0, this.oldItem);
		// // 更新this.List,觸發(fā)transition-group的動畫效果
		this.List = [...newList];
		this.list = [...newlist];
	}
}

完整代碼

gitee(碼云) - mj01分支 - dragAndDrop 文件


vue2+mousedown實現(xiàn)全屏拖動,全屏投擲

html

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>鼠標滑動</title>
    <link rel="stylesheet" href="./index.css">
</head>

<body>
	<div id="app">
        <div class="ctn ctn1">
            <div class="sub sub1" v-for="(site, index) in list1">
                <div class="dragCtn fixed" @mousedown="mousedown(site, $event)"
                    @mousemove.prevent='mousemove(site, $event)' @mouseup='mouseup(site, $event)'>
                    {{ site.name }}
                </div>
            </div>
        </div>
        <div class="ctn ctn2">
            <div class="sub sub2" v-for="(site, index) in list2">
                <div class="dragCtn">
                    {{ index }} : {{ site.name }}
                </div>
            </div>
        </div>
    </div>

    <script src="/node_modules/vue/dist/vue.js"></script>
    <script src="./index.js"></script>
</body>

</html>

JavaScript

new Vue({
    el: '#app',
    data: {
        list1: [{ name: '拖動我', index: 0 }],
        list2: [{ name: 'a', index: 0 }, { name: 'b', index: 1 }, { name: 'c', index: 2 }, { name: 'd', index: 3 }],
        vm: '',
        sb_bkx: 0,
        sb_bky: 0,
        is_moving: false
    },
    methods: {
        mousedown: function (site, event) {
            var startx = event.x;
            var starty = event.y;
            this.sb_bkx = startx - event.target.offsetLeft;
            this.sb_bky = starty - event.target.offsetTop;
            this.is_moving = true;
        },
        mousemove: function (site, event) {
            var endx = event.x - this.sb_bkx;
            var endy = event.y - this.sb_bky;
            var _this = this
            if (this.is_moving) {
                event.target.style.left = endx + 'px';
                event.target.style.top = endy + 'px';
            }
        },
        mouseup: function (e) {
            this.is_moving = false;
        }
    }
});

css

.ctn {
    line-height: 50px;
    cursor: pointer;
    font-size: 20px;
    text-align: center;
    float: left;
}

.sub:hover {
    background: #e6dcdc;
    color: white;
    width: 100px;
}

.ctn1 {
    border: 1px solid green;
    width: 100px;
}

.ctn2 {
    border: 1px solid black;
    width: 100px;
    margin-left: 50px;
}

.fixed {
    width: 100px;
    height: 100px;
    position: fixed;
    background: red;
    left: 10px;
    top: 10px;
    cursor: move;
}

vue+element+vuedraggable實現(xiàn)拖拽排序

使用yarn add vuedraggable或者npm i -S vuedraggable安裝拖拽組件。


<draggable
	v-model="codeList"
	@update="datadragEnd"
	:options="{ animation: 200 }"
>
	<div class="drag-item" v-for="(item, i) in codeList" :key="i">
	<el-row>
		<el-col class="line" :span="6">&nbsp;{{ item.field_title }}</el-col>
	</el-row>
	</div>
</draggable>

import draggable from "vuedraggable";

async datadragEnd(evt) {
	evt.preventDefault();
	// console.log('拖動前的索引 :' + evt.oldIndex)
	// console.log('拖動后的索引 :' + evt.newIndex)
	// 遍歷數(shù)組,將索引值賦值到對應(yīng)的 sort_order上面,完成排序
	let arr = this.codeList;
	let newArr = await arr.map((item, i) => {
		return {
			sort_order: i,
			field_code: item.field_code,
		};
	});
	const res = await this.$axios.post(`customer/save_order`, {
		list: newArr,
	});
	// console.log(res);
	const { error, message } = res.data;
	if (error == 0) {
		this.$message.success(message);
	}
},

vue3+element-plus+vuedraggable實現(xiàn)圖片上傳拖拽排序(若依)

前言

安裝對應(yīng)的vuedraggable組件
npm install vuedraggable@4.1.0 --save
package.json文件中記錄對應(yīng)的版本號為: "vuedraggable": "4.1.0",這里要注意咯?。?!克隆項目的時候這里的4.1.0可能會變?yōu)?code>^4.1.0,一定要改為4.1.0;如果不是可以先卸載然后安裝正確的版本即可。
如果版本不對會報錯,并且不能運行。


本案例基于若依vue3前后端分離項目做二次開發(fā)
若依自帶二次封裝element-plus圖片上傳組件,但是沒有實現(xiàn)拖拽排序功能。
于是又自己封裝了一個ImageUploadDraggable圖片上傳組件,此組件基于若依自帶的圖片上傳組件的基礎(chǔ)上進行再次封裝。
組件正常引入即可,可以全局引入或局部引入,引入方式跟我們自定的組件一樣。


html

<el-form-item label="圖片" class="ws_n">
  <image-upload-draggable v-model="dialogForm.images" :limit="5">
  </image-upload-draggable>
</el-form-item>

JavaScript

let info = reactive({
    dialogForm: {
      // 圖片
      images: []
    }
  }),
  {
    dialogForm
  } = toRefs(info);

二次封裝上傳組件

<template>
  <div class="component-upload-image">
    <ul class="el-upload-list el-upload-list--picture-card">
      <vue-draggable-next v-model="fileList">
        <li v-for="(item, index) in fileList" :key="item.index" class="el-upload-list__item is-success animated">
          <img :src="item.url" alt="" class="el-upload-list__item-thumbnail" />
          <i class="el-icon-close"></i>
          <span class="el-upload-list__item-actions">
            <!-- 預覽 -->
            <span class="el-upload-list__item-preview" @click="handlePictureCardPreviewFileDetail(item)">
              <el-icon>
                <zoom-in></zoom-in>
              </el-icon>
            </span>
            <!-- 刪除 -->
            <span class="el-upload-list__item-delete" @click="handleRemoveFileDetail(index)">
              <el-icon>
                <delete></delete>
              </el-icon>
            </span>
          </span>
        </li>
      </vue-draggable-next>
    </ul>
    <el-upload multiple :action="uploadImgUrl" list-type="picture-card" :on-success="handleUploadSuccess"
      :before-upload="handleBeforeUpload" :limit="limit" :on-error="handleUploadError" :on-exceed="handleExceed"
      ref="imageUpload" :show-file-list="false" :headers="headers" :class="{ hide: fileList.length >= limit }">
      <el-icon class="avatar-uploader-icon">
        <plus />
      </el-icon>
    </el-upload>
    <!-- 上傳提示 -->
    <div class="el-upload__tip" v-if="showTip">
      請上傳
      <template v-if="fileSize">
        大小不超過
        <b style="color: #f56c6c">{{ fileSize }}MB</b>
      </template>
      <template v-if="fileType">
        格式為
        <b style="color: #f56c6c">{{ fileType.join("/") }}</b>
      </template>
      的文件
    </div>

    <el-dialog v-model="dialogVisible" title="預覽" width="800px" append-to-body>
      <img :src="dialogImageUrl" style="display: block; max-width: 100%; margin: 0 auto" />
    </el-dialog>
  </div>
</template>

<script setup>
import { VueDraggableNext } from "vue-draggable-next";
import { getToken } from "@/utils/auth";

const props = defineProps({
  modelValue: [String, Object, Array],
  // 圖片數(shù)量限制
  limit: {
    type: Number,
    default: 5,
  },
  // 大小限制(MB)
  fileSize: {
    type: Number,
    default: 5,
  },
  // 文件類型, 例如['png', 'jpg', 'jpeg']
  fileType: {
    type: Array,
    default: () => ["png", "jpg", "jpeg"],
  },
  // 是否顯示提示
  isShowTip: {
    type: Boolean,
    default: true,
  },
});

const { proxy } = getCurrentInstance();
const emit = defineEmits();
const number = ref(0);
const uploadList = ref([]);
const dialogImageUrl = ref("");
const dialogVisible = ref(false);
const baseUrl = import.meta.env.VITE_APP_BASE_API;
// 上傳的圖片服務(wù)器地址
const uploadImgUrl = ref(import.meta.env.VITE_APP_BASE_API + "/file/upload");
const headers = ref({
  Authorization: "Bearer " + getToken(),
  appid: import.meta.env.VITE_APP_ID,
});
const fileList = ref([]);
const showTip = computed(
  () => props.isShowTip && (props.fileType || props.fileSize)
);

watch(
  () => props.modelValue,
  (val) => {
    if (val) {
      // 首先將值轉(zhuǎn)為數(shù)組
      const list = Array.isArray(val) ? val : props.modelValue.split(",");
      // 然后將數(shù)組轉(zhuǎn)為對象數(shù)組
      fileList.value = list.map((item) => {
        if (typeof item === "string") {
          item = { name: item, url: item };
        }
        return item;
      });
    } else {
      fileList.value = [];
      return [];
    }
  },
  { deep: true, immediate: true }
);

// 上傳前l(fā)oading加載
function handleBeforeUpload(file) {
  let isImg = false;
  if (props.fileType.length) {
    let fileExtension = "";
    if (file.name.lastIndexOf(".") > -1) {
      fileExtension = file.name.slice(file.name.lastIndexOf(".") + 1);
    }
    isImg = props.fileType.some((type) => {
      if (file.type.indexOf(type) > -1) return true;
      if (fileExtension && fileExtension.indexOf(type) > -1) return true;
      return false;
    });
  } else {
    isImg = file.type.indexOf("image") > -1;
  }
  if (!isImg) {
    proxy.$modal.msgError(
      `文件格式不正確, 請上傳${props.fileType.join("/")}圖片格式文件!`
    );
    return false;
  }
  if (props.fileSize) {
    const isLt = file.size / 1024 / 1024 < props.fileSize;
    if (!isLt) {
      proxy.$modal.msgError(`上傳頭像圖片大小不能超過 ${props.fileSize} MB!`);
      return false;
    }
  }
  proxy.$modal.loading("正在上傳圖片,請稍候...");
  number.value++;
}

// 文件個數(shù)超出
function handleExceed() {
  proxy.$modal.msgError(`上傳文件數(shù)量不能超過 ${props.limit} 個!`);
}

// 上傳成功回調(diào)
function handleUploadSuccess(res, file) {
  if (res.code === 0) {
    uploadList.value.push({ name: res.data.url, url: res.data.url });
    uploadedSuccessfully();
  } else {
    number.value--;
    proxy.$modal.closeLoading();
    proxy.$modal.msgError(res.msg);
    proxy.$refs.imageUpload.handleRemove(file);
    uploadedSuccessfully();
  }
}

function handlePictureCardPreviewFileDetail(file) {
  dialogImageUrl.value = file.url;
  dialogVisible.value = true;
}

// 刪除
function handleRemoveFileDetail(index) {
  fileList.value.splice(index, 1);
}

// 上傳結(jié)束處理
function uploadedSuccessfully() {
  if (number.value > 0 && uploadList.value.length === number.value) {
    fileList.value = fileList.value
      .filter((f) => f.url !== undefined)
      .concat(uploadList.value);
    uploadList.value = [];
    number.value = 0;
    emit("update:modelValue", listToString(fileList.value));
    proxy.$modal.closeLoading();
  }
}
// 上傳失敗
function handleUploadError() {
  proxy.$modal.msgError("上傳圖片失敗");
  proxy.$modal.closeLoading();
}

// 對象轉(zhuǎn)成指定字符串分隔
function listToString(list, separator) {
  let strs = "";
  separator = separator || ",";
  for (let i in list) {
    if (undefined !== list[i].url && list[i].url.indexOf("blob:") !== 0) {
      strs += list[i].url.replace(baseUrl, "") + separator;
    }
  }
  return strs != "" ? strs.substr(0, strs.length - 1) : "";
}
</script>

<style scoped lang="scss">
// .el-upload--picture-card 控制加號部分
:deep(.hide .el-upload--picture-card) {
  display: none;
}
</style>

vue2+transition-group實現(xiàn)拖動排序

html

<transition-group id='app' name="drog" tag="ul">
	<div draggable="true" v-for="(item, index) in lists" @dragstart="dragStart($event, index)" @dragover="allowDrop" @drop="drop($event, index)" v-bind:key="item">{{item}}</div>
</transition-group>

JavaScript

new Vue({
    el: '#app',
    data: {
        lists: ['1: apple', '2: banana', '3: orange', '4: melon']
    },
    
	methods: {
        // 取消默認行為
        allowDrop(e){
            e.preventDefault();
        },
        
        // 開始拖動
        dragStart(e, index){
            let tar = e.target;
            e.dataTransfer.setData('Text', index);
            if (tar.tagName.toLowerCase() == 'li') {
                // console.log('drag start')
                // console.log('drag Index: ' + index)
            }
        },
        
        // 放置
        drop(e, index){
            this.allowDrop(e);
            // console.log('drop index: ' + index);
            //使用一個新數(shù)組重新排序后賦給原變量
            let arr = this.lists.concat([]),
                dragIndex = e.dataTransfer.getData('Text');
                temp = arr.splice(dragIndex, 1);
            arr.splice(index, 0, temp[0]);
            // console.log('sort');
            this.lists = arr;
        }
    }
});

原生拖拽排序

html

<ul id="idUl">
    <li class="m_36 ta_c bc_87ceeb fs_68">1</li>
    <li class="m_36 ta_c bc_87ceeb fs_68">2</li>
    <li class="m_36 ta_c bc_87ceeb fs_68">3</li>
    <li class="m_36 ta_c bc_87ceeb fs_68">4</li>
    <li class="m_36 ta_c bc_87ceeb fs_68">5</li>
</ul>

JavaScript文章來源地址http://www.zghlxwxcb.cn/news/detail-813392.html

(function () {
    let ulList = document.querySelector('#idUl'),
        liList = document.querySelectorAll('li'),
        currentLi = undefined;

    liList.forEach(item => item.draggable = "true");

    ulList.addEventListener('dragstart', (e) => {
        e.dataTransfer.effectAllowed = 'move';
        currentLi = e.target;

        setTimeout(() => currentLi.classList.add('bc_transparent color_transparent'), 0);
    });
    ulList.addEventListener('dragenter', (e) => {
        e.preventDefault();

        if (e.target === currentLi || e.target === ulList) return false;

        let liArray = Array.from(ulList.childNodes),
            currentIndex = liArray.indexOf(currentLi),
            targetindex = liArray.indexOf(e.target)

        if (currentIndex < targetindex) {
            ulList.insertBefore(currentLi, e.target.nextElementSibling);
        } else {
            ulList.insertBefore(currentLi, e.target);
        }
    });
    ulList.addEventListener('dragover', (e) => e.preventDefault());
    ulList.addEventListener('dragend', (e) => currentLi.classList.remove('bc_transparent color_transparent'));
})();

到了這里,關(guān)于web前端之拖拽API、上傳多圖片時拖拽排序、表格行或列拖拽排序、復制元素跨區(qū)域放置、拖放、投擲、若依、vuedraggable、sortablejs、element、plus、vue、ui的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來自互聯(lián)網(wǎng)用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務(wù),不擁有所有權(quán),不承擔相關(guān)法律責任。如若轉(zhuǎn)載,請注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實不符,請點擊違法舉報進行投訴反饋,一經(jīng)查實,立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費用

相關(guān)文章

  • 微信小程序單圖上傳和多圖上傳

    微信小程序單圖上傳和多圖上傳

    圖片上傳主要用到 1、wx.chooseImage(Object object) 從本地相冊選擇圖片或使用相機拍照。 Object object 屬性 類型 默認值 必填 說明 count number 9 否 最多可以選擇的圖片張數(shù) sizeType Array.string [\\\'original\\\', \\\'compressed\\\'] 否 所選的圖片的尺寸 sourceType Array.string [\\\'album\\\', \\\'camera\\\'] 否 選擇圖片的來源

    2024年02月03日
    瀏覽(24)
  • 前端學習——Web API(Day1)

    前端學習——Web API(Day1)

    Web API 基本認知 作用和分類 DOM DOM樹 DOM對象 根據(jù)CSS選擇器來獲取DOM元素 小練習 其他獲取DOM元素方法 識別標簽 小練習 操作元素常用屬性 小練習 操作元素樣式屬性 小練習 小練習 操作表單元素屬性 自定義屬性 小練習

    2024年02月13日
    瀏覽(18)
  • layui選擇多張圖片上傳多圖上傳到服務(wù)器保存

    多圖上傳在一些特殊的需求中我們經(jīng)常會遇到,其實多圖上傳的原理大家都有各自的見解。對于Layui多圖上傳和我之前所說的通過js獲取文本框中的文件數(shù)組遍歷提交的原理一樣,只不過是Layui中的upload.render方法已經(jīng)幫我們封裝好了,我們只管調(diào)用即可,也就是說你選中了幾張

    2024年02月16日
    瀏覽(15)
  • 原生JavaScript+PHP多圖上傳實現(xiàn)

    原生JavaScript+PHP多圖上傳實現(xiàn)

    很多場景下需要選擇多張圖片上傳,或者是批量上傳以提高效率,多圖上傳的需求自然就比較多了,本文使用最簡單的XMLHttpRequest異步上傳圖片。 index.html upload.php (請建立一個upload文件夾以存放上傳的文件) TANKING

    2024年02月09日
    瀏覽(25)
  • 【前端】使用Web Audio API 技術(shù)播放音樂

    【前端】使用Web Audio API 技術(shù)播放音樂

    記錄下使用web audio播放音樂的方法。 Web Audio API 提供了在 Web 上控制音頻的一個非常有效通用的系統(tǒng),允許開發(fā)者來自選音頻源,對音頻添加特效,使音頻可視化,添加空間效果(如平移),等等。 你可以先看下api接口介紹文章Web Audio API接口介紹。 html 的 audio已經(jīng)可以播放音

    2024年04月25日
    瀏覽(19)
  • uniapp、uview——圖片上傳(單圖上傳、多圖上傳、多組照片上傳、圖片回顯)

    uniapp、uview——圖片上傳(單圖上傳、多圖上傳、多組照片上傳、圖片回顯)

    uView組件的上傳功能,單圖上傳、多圖上傳等。 官方文檔地址: https://www.uviewui.com/components/upload.html (一)單圖上傳 1.效果演示: 只能上傳一張,選完之后,上傳的按鈕消失,當然,如果圖片不合適,刪掉再換一張,但就是只能上傳一張。 2.代碼: (二)多圖上傳 1.效果演

    2024年02月07日
    瀏覽(21)
  • Flask Web API構(gòu)建實例:GET、POST文件上傳、靜態(tài)資源下載一網(wǎng)打盡

    Flask Web API構(gòu)建實例:GET、POST文件上傳、靜態(tài)資源下載一網(wǎng)打盡

    ? 以下是一個通過 Flask 構(gòu)建 Web API 服務(wù)的詳細示例,包含了各類請求(GET、POST、文件上傳、靜態(tài)資源下載)、每個方法獨立配置路由、參數(shù)接收和解析、請求日志記錄以及異常日志記錄。請確保你已經(jīng)安裝了 Flask,你可以使用以下命令進行安裝: 接下來是一個包含詳細注釋

    2024年02月04日
    瀏覽(20)
  • thinkadmin表單上傳單圖,多圖,單文件,多文件
  • .net6Api后臺+VUE3前端實現(xiàn)上傳和下載文件全過程

    .net6Api后臺+VUE3前端實現(xiàn)上傳和下載文件全過程

    首先本文參考的是,感謝博主: net6WebApi上傳下載文件_cduoa的博客-CSDN博客_webapi下載文件 在博主的基礎(chǔ)上,增加了新的功能,代碼中有注明,并且使用VUE3前端實現(xiàn)。 后端部分: 1.首先建立IFileService文件 2.建立FileService文件 3.增加FileController文件 4.Program文件中,進行配置和跨域

    2023年04月09日
    瀏覽(33)
  • 前端調(diào)用 Stable diffusion 秋葉啟動器 Web-Ui API

    前端調(diào)用 Stable diffusion 秋葉啟動器 Web-Ui API

    在啟動器啟動前,開啟啟用 API 就可以調(diào)用 通過訪問 API DOCS 查看完整的接口地址。 POST 請求體以 JSON 形式發(fā)送。 以 axios 為例: 文生圖 地址:/sdapi/v1/txt2img 參數(shù)說明 圖生圖 地址:/sdapi/v1/img2img 其中 init_images 參數(shù)可以是服務(wù)文件路徑也可以是 base64

    2024年02月08日
    瀏覽(26)

覺得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包