中文亚洲精品无码_熟女乱子伦免费_人人超碰人人爱国产_亚洲熟妇女综合网

當前位置: 首頁 > news >正文

網(wǎng)站網(wǎng)頁設計要求真正免費的網(wǎng)站建站平臺運營

網(wǎng)站網(wǎng)頁設計要求,真正免費的網(wǎng)站建站平臺運營,做網(wǎng)站不會寫代碼,最近上海大事件最近碰到了個需求,大概就是要通過可視化拖拽的方式配置一個冰柜,需要把預設好的冰柜內部架子模板一個個拖到冰箱內。一開始的想法是用鼠標事件(mousedown、mouseup等)那一套去實現(xiàn),能實現(xiàn)但是過程過于復雜,…

最近碰到了個需求,大概就是要通過可視化拖拽的方式配置一個冰柜,需要把預設好的冰柜內部架子模板一個個拖到冰箱內。一開始的想法是用鼠標事件(mousedown、mouseup等)那一套去實現(xiàn),能實現(xiàn)但是過程過于復雜,需要控制的狀態(tài)太多了。其實 Web Api 為 html 元素拖拽量身定制了一套 HTML 拖放API,用這個方法實現(xiàn)一些簡單的拖拽功能簡直不要太簡單。為此寫了這篇文章,下面將詳細介紹 HTML 拖放 API 的核心知識點

文檔

一、被拖拽元素和放置被拖拽元素的元素

通常我們所了解的拖放是按住鼠標左鍵不放然后移動鼠標把一個頁面元素從某個位置移動到另一個位置,然后松開鼠標左鍵,至此完成了整個拖放過程。在這個過程中我們需要先重點關注兩個東西,一個是被拖拽元素另一個是 放置被拖拽元素的元素。

1.1 被拖拽元素

我們得先有個概念,頁面上顯示的元素默認并不都是可以被拖拽的(除了圖片、被選中的文字、鏈接),所以如果當前元素默認不可被拖拽那么就得先把它設置為可拖拽的。ps:可拖拽元素被拖拽時會有一個半透明的快照跟著鼠標移動。

將 HTML 元素的 draggable 屬性設置為 true, 元素就可以變?yōu)榭赏献г?。效果如下圖。

<div id="box" draggable="true">draggable box</div>

在這里插入圖片描述

1.2 可放置被拖拽元素的元素

所有的元素區(qū)域默認是不支持放置被拖拽元素的,直觀的表現(xiàn)是,當被拖拽元素經(jīng)過不可放置區(qū)域時鼠標的樣式是一個禁止放置的一個圖標(圓圈帶一個斜杠),所以需要將目標元素設置為一個可放置區(qū)域

默認情況下是這樣:

<div id="box" draggable="true">draggable box</div>
<div id="droppable">放置區(qū)域</div>

在這里插入圖片描述
設置為放置區(qū)域需要給元素綁定一個事件 dragover 且要 阻止默認行為

<div id="box" draggable="true">draggable box</div>
<div id="droppable">放置區(qū)域</div>
<script>let dropDom = document.getElementById('droppable')dropDom.addEventListener('dragover', (e) => {e.preventDefault()})
</script>

React:
放置區(qū)域需要綁定onDragOver事件,且要 阻止默認行為 – 其他事件一樣加on

<div draggable="true">draggable box</div>
<div onDragOver={(e) => {e.preventDefault()}}>放置區(qū)域</div>

設置為可放置區(qū)域后鼠標樣式也變了不再是禁止圖標,而是一個加號圖標(圖標可以設置,下面會講解):
在這里插入圖片描述
然而你會發(fā)現(xiàn)被拖放元素并沒有真正的被放置到放置區(qū)域,這是必然的,放置操作需要開發(fā)者自行定義,以上的設置只是是為了向用戶表明這個區(qū)域是允許放東西的,那么至于怎么放需要開發(fā)者自行決定。

二、拖拽過程觸發(fā)的一些事件

這一小節(jié)將帶你了解整個拖放過程的其他細節(jié),比如拖拽過程中會觸發(fā)哪些事件

2.1 被拖放目標觸發(fā)的事件

給被拖放目標元素綁定三個事件 dragstart、drag、dragend。

<div id="box" draggable="true">draggable box</div>
<div id="droppable">放置區(qū)域</div>
<script>let dragDom = document.getElementById('box')dragDom.addEventListener('dragstart', (e) => {console.log('開始拖動');})dragDom.addEventListener('drag', (e) => {console.log('拖動中');})dragDom.addEventListener('dragend', (e) => {console.log('結束拖動');})
</script>

React

<div  draggable="true"onDragStart={(e) => {console.log("開始拖動", e);}}onDrag={(e) => {console.log("拖動中", e);}}onDragEnd={(e) => {console.log("結束拖動", e);}}
>
>draggable box</div>
<div onDragOver={(e) => {e.preventDefault()}}>放置區(qū)域</div>

開始拖動觸發(fā) dragstart ,拖動過程中(鼠標不松開)觸發(fā)drag,松開鼠標(或者按下 Esc 鍵)觸發(fā) dragend。
在這里插入圖片描述

2.2 被拖拽元素在放置區(qū)域內會觸發(fā)的事件

先給放置目標元素綁定四個事件

<div id="box" draggable="true">draggable box</div>
<div id="droppable">放置區(qū)域</div>
<script>let dropDom = document.getElementById('droppable')dropDom.addEventListener('dragenter', (e) => {console.log('進入到了放置區(qū)域~');})dropDom.addEventListener('dragover', (e) => {e.preventDefault()console.log('在放置區(qū)域內拖拽中~');})dropDom.addEventListener('dragleave', (e) => {console.log('離開了放置區(qū)域~');})dropDom.addEventListener('drop', (e) => {console.log('在放置區(qū)域內,放下了被拖拽元素~')})
</script>

拖拽元素進入放置區(qū)域內時觸發(fā) dragenter 事件,在放置區(qū)域內移動被拖放(鼠標不松開)元素觸發(fā) dragover 事件,被拖放元素離開放置區(qū)域觸發(fā) dragleave 事件,在放置區(qū)域內松開鼠標觸發(fā) drop 事件。
在這里插入圖片描述

三、實現(xiàn)真正意義上的元素拖放

通過上面觸發(fā)的事件我們可以知道,用戶真正在放置區(qū)域釋放鼠標的時候只有 drop 事件能夠監(jiān)聽到。所以開發(fā)者需要在這個事件里做真正的放置操作,放置什么由開發(fā)者決定,可以是被拖拽元素,也可以是自定義的一些內容。

放置被拖拽元素:

<div id="box" draggable="true">draggable box</div>
<div id="droppable">放置區(qū)域</div>
<script>let dropDom = document.getElementById('droppable')dropDom.addEventListener('dragover', (e) => {e.preventDefault()console.log('在放置區(qū)域內拖拽中~');})dropDom.addEventListener('drop', (e) => {console.log('在放置區(qū)域內,放下了被拖拽元素~')e.target.appendChild(document.getElementById('box'))})
</script>

在這里插入圖片描述
放置自定義內容

dropDom.addEventListener('drop', (e) => {console.log('在放置區(qū)域內,放下了被拖拽元素~')let customCOntent = '<p>自定義內容</p>'e.target.innerHTML = e.target.innerHTML + customCOntent
})

在這里插入圖片描述

四、dataTransfer 對象

4.1 從被拖放元素向可放置元素傳遞數(shù)據(jù)

dataTransfer 對象提供了一個setData()方法,它接受兩個參數(shù),第一個參數(shù)是傳遞數(shù)據(jù)的類型(一般是標準的MIME類型),第二個數(shù)據(jù)是數(shù)據(jù)值。dataTransfer 還提供了 getData() 的方法用于獲取傳遞的數(shù)據(jù),它接受一個參數(shù),參數(shù)值為 setData 對應的第一個參數(shù)。

傳遞一個簡單的字符串數(shù)據(jù)

<div id="box" draggable="true">draggable box</div>
<div id="droppable">放置區(qū)域</div>
<script>let dropDom = document.getElementById('droppable')let dragDom = document.getElementById('box')dragDom.addEventListener('dragstart', (e) => {e.dataTransfer.setData('text/plain', '自定義數(shù)據(jù)')})dropDom.addEventListener('dragover', (e) => {e.preventDefault()})dropDom.addEventListener('drop', (e) => {let data = e.dataTransfer.getData('text/plain')console.log('你傳遞的數(shù)據(jù)為:', data);})
</script>

在這里插入圖片描述
?注意:只能在 dragstart 事件中設置數(shù)據(jù),在其他地方設置無效。且只能在 drop 事件中獲取設置的數(shù)據(jù),其他事件中獲取不到。
案例:根據(jù)傳遞的數(shù)據(jù)放置不同的內容。

<div id="box" draggable="true">draggable box</div>
<div id="droppable">放置區(qū)域</div>
<script>let dropDom = document.getElementById('droppable')let dragDom = document.getElementById('box')dropDom.addEventListener('dragover', (e) => {e.preventDefault()})dropDom.addEventListener('drop', (e) => {let num = e.dataTransfer.getData('num')console.log(num);if(num > 5)e.target.innerHTML = e.target.innerHTML + '<p>傳遞的數(shù)字大于5</p>'else if(num == 5) e.target.innerHTML = e.target.innerHTML + '<p>傳遞的數(shù)字等于5</p>'elsee.target.innerHTML = e.target.innerHTML + '<p>傳遞的數(shù)字小于5</p>'})dragDom.addEventListener('dragstart', (e) => {let num = Math.floor(Math.random() * 10) + 1;e.dataTransfer.setData('num', num)})
</script>

在這里插入圖片描述

4.2 自定義拖拽過程中跟隨鼠標移動的內容

默認情況下元素被拖拽時會有一個半透明的元素快照跟隨著鼠標移動。通過 dataTransfer 提供的 setDragImage(elemnt, xOffset, yOffset) 方法是可以自定義跟隨內容。接受三個參數(shù) elemnt 可以是 dom 節(jié)點或者一個圖片對象,xOffset, yOffset 是相對于鼠標的偏移量。

語法

dataTransfer.setDragImage(img, xOffset, yOffset);

img | Element
用于拖曳反饋圖像的圖像 Element 元素。

如果 Element 是一個 img 元素,則將拖動位圖設置為該元素的圖像(保持大小);否則,將拖動數(shù)位圖設置為從給定元素所生成的圖片

xOffset
使用 long 指示相對于圖片的橫向偏移量

yOffset
使用 long 指示相對于圖片的縱向偏移量

解析
  • 發(fā)生拖動時,從拖動目標 (dragstart事件觸發(fā)的元素) 生成半透明圖像,并在拖動過程中跟隨鼠標指針。這個圖片是自動創(chuàng)建的,你不需要自己去創(chuàng)建它。然而,如果想要設置為自定義圖像,那么 DataTransfer.setDragImage() 方法就能派上用場。
  • 圖像通常是一個 <image> 元素,但也可以是<canvas> 或任何其他圖像元素。該方法的 x 和 y 坐標是圖像應該相對于鼠標指針出現(xiàn)的偏移量。
    坐標指定鼠標指針相對于圖片的偏移量。例如,要使圖像居中,請使用圖像寬度和高度的一半。通常在 dragstart 事件處理程序中調用此方法。
實際用例

setDragImage 的第一個參數(shù)接受的是一個Element參數(shù),這樣的話,普通的html元素、image元素、canvas都可以傳遞

1、設置為一個圖片:

<script>import Tag from "../../style/imgs/attributeTag/路徑.png"; //已經(jīng)存在的圖片let dragDom = document.getElementById('box')dragDom.addEventListener('dragstart', (e) => {let img = new Image()// 創(chuàng)建一個圖像并且使用它作為拖動圖像// 請注意: 改變 "example.gif" 為一個已經(jīng)存在的圖片// 或者,一個還沒有創(chuàng)建出來的圖片,那么瀏覽器將會使用默認的拖動圖片// 譯者注:默認的拖動圖片與拖動對象沒有聯(lián)系。一般是一個小型文件圖標// 例如:// mg.src = Tag //或// mg.src = ``;img.src = 'example.gif'e.dataTransfer.setDragImage(img, 10, 10)})
</script>

在這里插入圖片描述
2、以官網(wǎng)例子為例,把canvas作為參數(shù)傳遞,我首先嘗試的是這種方式,發(fā)現(xiàn)并不能生效。(官方的例子沒有運行成功)

function dragWithCustomImage(event) {var canvas = document.createElementNS("http://www.w3.org/1999/xhtml","canvas");canvas.width = canvas.height = 50;var ctx = canvas.getContext("2d");ctx.lineWidth = 4;ctx.moveTo(0, 0);ctx.lineTo(50, 50);ctx.moveTo(0, 50);ctx.lineTo(50, 0);ctx.stroke();var dt = event.dataTransfer;dt.setData('text/plain', 'Data to Drag');dt.setDragImage(canvas, 25, 25);
}

3、根據(jù)案例,我接著使用·HtmlDivElement·作為參數(shù)傳遞,創(chuàng)建了·DIV元素·,此時也·沒有生效·。

export function drawDragImage(dataTransfer: DataTransfer, context: string) {
const drawItem: API.EquipmentInfo = JSON.parse(context);
const div = document.createElement(‘div’);
div.style.height = itemObj.height + ‘px’;
div.style.width = itemObj.width + ‘px’;
div.style.border = ‘1px solid #000’;
const span = document.createElement(‘span’);
span.innerText = ‘2222’;
div.appendChild(span);
dataTransfer.setDragImage(div, drawItem.width / 2, drawItem.height / 2);
}

4、然后,我改進了canvas,把canvas轉化為圖片,第一次拖拽的時候,因為image加載元素是異步導致了沒有生效,如圖1;第二以后拖拽的時候可以生效,如圖二。

  const imageContent = canvas.toDataURL('image/jpeg', 1);const image = new Image();image.src = imageContent;image.onload = () => {console.log('image2 load');};dataTransfer.setDragImage(image, drawItem.width / 2, drawItem.height / 2);

在這里插入圖片描述

5、最后我嘗試了使用官網(wǎng)的方法,同樣因為image加載圖片是異步的,而拖拽事件是同步發(fā)生的,導致了第一次執(zhí)行失敗。

function dragstart_handler(ev) {console.log("dragStart");// 設置拖動的格式和數(shù)據(jù)。使用事件目標的 id 作為數(shù)據(jù)ev.dataTransfer.setData("text/plain", ev.target.id);// 創(chuàng)建一個圖像并且使用它作為拖動圖像// 請注意:改變 "example.gif" 為一個已經(jīng)存在的圖片// 或者,一個還沒有創(chuàng)建出來的圖片,那么瀏覽器將會使用默認的拖動圖片// 譯者注:默認的拖動圖片與拖動對象沒有聯(lián)系。一般是一個小型文件圖標var img = new Image();img.src = 'example.gif';ev.dataTransfer.setDragImage(img, 10, 10);
}
解決方案

在嘗試了不同方式設置拖拽反饋圖像,總結了一些解決方案

  • 以html頁面的元素為模版,動態(tài)生成內容,然后設置Element元素參數(shù),可以設置DIV元素的z-index(使用z-index,必須使用position:relative | absolute)–(嘗試使用過css1、position:absolute 定位出瀏覽器可視界面 2、 display:none無用),隱藏在實際頁面之下:這樣可以動態(tài)生成要拖拽的元素,并和生成的fabric的group保持一致。 完美的解決了問題 。
  • 在這里插入圖片描述
    js
export function drawDragImage(dataTransfer: DataTransfer, context: string) {const drawItem: API.EquipmentInfo = JSON.parse(context);const dragElement = document.getElementById('dragItem');const idElement = dragElement?.getElementsByClassName('dragItemId')[0];const nameElement = dragElement?.getElementsByClassName('dragItemName')[0];if (idElement) {idElement.innerHTML = drawItem.id;}if (nameElement) {nameElement.innerHTML = drawItem.typeName || '';}if (dragElement) {dragElement.style.height = drawItem.height + 'px';dragElement.style.width = drawItem.width + 'px';dragElement.style.border = '1px solid #000';dragElement.style.background = '#fff';}if (dragElement) {dataTransfer.setDragImage(dragElement, drawItem.width / 2, drawItem.height / 2);}}

React

 import {useRef} from "react";import { Modal, Space, Input, Tree, Button, Badge } from "antd";const mouseStyle = useRef<any>(null);<divdraggable="true"onDragStart={(e) => {//mouseStyle.currente.dataTransfer.setDragImage(mouseStyle.current, 10, 10);}}>移動位置</div>//	
//absolute top-[10%] z-[1] h-10    css使用了tailwindcss
//npm install -D tailwindcss
//https://tailwindcss.com/docs/installation 文檔地址<div className="absolute top-[10%] z-[1] h-10" ref={mouseStyle}><Badge count={5}><div className=" border border-[#444444] leading-6 h-6 w-15">2023.09.22初級會計資格考試</div></Badge>
</div>

在這里插入圖片描述

4.3 設置放置前的反饋圖標

dataTransfer 提供了一個 dropEffect屬性設置放置前的反饋圖標,它有四種取值 none move copy link

在 dragover 中設置 dropEffect 的值

dropDom.addEventListener('dragover', (e) => {e.preventDefault()e.dataTransfer.dropEffect = 'link' // none || move || copy || link
})
  • 值為 none 或者經(jīng)過不可放置區(qū)域,顯示禁止放置圖標在這里插入圖片描述
  • 值為 move 時
    在這里插入圖片描述
  • 值為 copy 時
    在這里插入圖片描述
  • 值為 link 時
    在這里插入圖片描述

4.4 拖動文件上傳

通過 dataTransfer 的 files 屬性可以獲取用戶拖拽的文件信息

拖拽系統(tǒng)文件到放置區(qū)域,并打印拖拽的文件信息:

dropDom.addEventListener('drop', (e) => {e.preventDefault()// 上傳的文件列表let fileList = e.dataTransfer.filesfor (let i = 0; i < fileList.length; i++) {const file = fileList[i];console.log('文件名:' + file.name);console.log('文件大小:' + file.size);// 后續(xù)操作 比如:調接口上傳文件}
})

在這里插入圖片描述

4.5 清除 setData() 的值

dataTransfer 提供了 clearData() 清除 setData 設置的值,傳參數(shù)則刪除指定類型的值,不傳則全部清除。

dropDom.addEventListener('drop', (e) => {console.log(e.dataTransfer.getData('text/plain'));console.log(e.dataTransfer.getData('text/html'));
})dragDom.addEventListener('dragstart', (e) => {e.dataTransfer.setData('text/plain', '自定義數(shù)據(jù)')e.dataTransfer.setData('text/html', '自定義數(shù)據(jù)2')e.dataTransfer.clearData('text/html')
})

在這里插入圖片描述

4.6 查看設置了哪些類型的值

dataTransfer 提供了 types 屬性查看 setData 設置了哪些類型的值。

dropDom.addEventListener('drop', (e) => {console.log(e.dataTransfer.types);
})
dragDom.addEventListener('dragstart', (e) => {e.dataTransfer.setData('text/plain', '自定義數(shù)據(jù)')e.dataTransfer.setData('text/html', '自定義數(shù)據(jù)2')
})

在這里插入圖片描述

4.7 effectAllowed 屬性取值會影響到 dropEffect 的取值效果。

effectAllowed 用于限制 dropEffect 只能設置哪些值

effectAllowed 的取值有: + none -> 此項表示 dropEffect 設置任何值都是禁止效果 + copy -> dropEffect 可以設置為 copy + copyLink -> dropEffect 可以設置為 copy 和 link + copyMove -> dropEffect 可以設置為 copy 和 Move + link -> dropEffect 可以設置為 link + linkMove -> dropEffect 可以設置為 link 和 Move + move -> dropEffect 可以設置為 Move + all -> dropEffect 可以設置為所有合法值 + uninitialized -> 等同 all 效果

dropDom.addEventListener('dragover', (e) => {e.preventDefault()e.dataTransfer.dropEffect = 'move'
})
dragDom.addEventListener('dragstart', (e) => {e.dataTransfer.effectAllowed = 'none'
})

上面即使 dropEffect 設置為 move, 但是 effectAllowed 的值為 none,所有還是禁止放置的反饋圖標。
在這里插入圖片描述

五、總結

  • 實現(xiàn)一個拖拽功能時先定義好被拖拽元素和放置區(qū)域元素。
  • 所有的放置操作都是在 drop 事件中完成。
  • 放置前的反饋效果可以根據(jù)你傳遞的數(shù)據(jù)來設置 dropEffect 顯示不同的效果。
  • 被拖拽元素也可以是放置區(qū)域,放置區(qū)域也可以是被拖拽元素,兩者沒有明確的界限。
  • 功能自定義按需求開發(fā)
http://www.risenshineclean.com/news/28694.html

相關文章:

  • 做五金有哪些網(wǎng)站推廣觀看b站的廣告網(wǎng)站平臺
  • 網(wǎng)站做淘寶推廣收入平臺seo什么意思
  • 網(wǎng)站怎樣在360做優(yōu)化溫州seo推廣外包
  • 政府網(wǎng)站建設方案范文 工作方案寰宇seo
  • 網(wǎng)站設計要多少錢網(wǎng)絡營銷培訓班
  • 做網(wǎng)站軟件要錢嗎都有什么推廣平臺
  • 知名網(wǎng)站開發(fā)語言成都網(wǎng)絡推廣哪家好
  • 網(wǎng)站怎么做最省錢百度資源搜索
  • 推廣網(wǎng)站seo廈門推廣平臺較好的
  • 建筑設計網(wǎng)站網(wǎng)址外鏈火
  • 景安網(wǎng)站備案的服務碼百度網(wǎng)絡營銷app
  • 注冊城鄉(xiāng)規(guī)劃師難度優(yōu)化網(wǎng)站打開速度
  • asp.net 網(wǎng)站管理工具 安全營銷軟件
  • 安徽網(wǎng)站開發(fā)費用做谷歌推廣比較好的公司
  • wordpress建的網(wǎng)站打開太慢優(yōu)化大師win10能用嗎
  • 域名怎么和網(wǎng)站綁定深圳網(wǎng)站快速排名優(yōu)化
  • 動態(tài)網(wǎng)站開發(fā)在線測試第5章策劃方案
  • 城建設投資公司網(wǎng)站最近國內新聞
  • 導航網(wǎng)站 php煙臺網(wǎng)絡推廣
  • 做跨境網(wǎng)站注意事項小白如何學電商運營
  • 武漢網(wǎng)站設計制作公司哪家好搜索引擎優(yōu)化時營銷關鍵詞
  • 神碼ai智能寫作網(wǎng)站百度怎么發(fā)帖做推廣
  • 什么樣的網(wǎng)站需要icp經(jīng)營性備案產(chǎn)品市場營銷策劃書
  • 自己做網(wǎng)站需要買什么手機怎么建自己的網(wǎng)站
  • 企業(yè)網(wǎng)站建設定制南寧求介紹seo軟件
  • 做網(wǎng)站關鍵詞網(wǎng)絡營銷的效果是什么
  • 做網(wǎng)站的注意什么國內最新消息新聞
  • 蚌埠哪里做網(wǎng)站站長權重
  • 南寧中小企業(yè)網(wǎng)站制作許昌seo公司
  • 管理網(wǎng)站開發(fā)教程semseo是什么意思