Blazor 使用拖放(drag and drop)上傳文件

来源:https://www.cnblogs.com/densen2014/archive/2022/04/11/16128246.html
-Advertisement-
Play Games

在很多上傳文件的應用實例中, 都可以看到[拖放文件到此上傳]這種騷功能 ,今天我們就來試試Blazor能不能完成這個想法. 原文鏈接:https://www.cnblogs.com/ysmc/p/16128206.html 簡述HTML5拖放 拖放是HTML5標準的一部分,任何元素都能夠拖放,也能夠 ...


在很多上傳文件的應用實例中, 都可以看到[拖放文件到此上傳]這種騷功能 ,今天我們就來試試Blazor能不能完成這個想法.

原文鏈接:https://www.cnblogs.com/ysmc/p/16128206.html  

簡述HTML5拖放

拖放是HTML5標準的一部分,任何元素都能夠拖放,也能夠將本地的文件拖放到頁面上。

設置元素可拖放

為了使元素可拖動,把 draggable 屬性設置為 true

<img draggable="true" />

拖放事件

有7個拖放事件可以捕獲,如下:

dragstart:開始拖元素觸發

dragenter:元素拖進可drop元素(綁定drop事件的元素)時觸發

dragover:當元素拖動到drop元素上時觸發

drop:當元素放下到drop元素觸發

dragleave :當元素離開drop元素時觸發

drag:每次元素被拖動時會觸發

dragend:放開拖動元素時觸發

完成一次拖放的事件過程是: dragstart –> dragenter –> dragover –> drop –> dragend

瀏覽器支持

Edge、Firefox、Opera 12、Chrome 以及 Safari 5 支持拖放。

拖拽上傳實現

1.新建工程n02drag,將項目添加到解決方案中

dotnet new blazorserver -o n02drag
dotnet sln add n02drag/n02drag.csproj

2.在文件夾wwwroot/lib,添加drag子文件夾,新建app.js文件夾

先阻止頁面預設的拖放行為,不然頁面會被拖放的文件替換了。

//阻止瀏覽器預設行為
document.addEventListener( "dragleave", function(e) {
     e.preventDefault();
}, false);
document.addEventListener( "drop", function(e) {
     e.preventDefault();
}, false);
document.addEventListener( "dragenter", function(e) {
     e.preventDefault();
}, false);
document.addEventListener( "dragover", function(e) {
     e.preventDefault();
}, false);

設置drop區域

當文件拖放到drop區域時,就能觸發上傳。

    element.addEventListener("drop", function (e) {

        try {
            var fileList = e.dataTransfer.files; //獲取文件對象
            //檢測是否是拖拽文件到頁面的操作
            if (fileList.length == 0) {
                return false;
            }

            inputFile.files = e.dataTransfer.files;
            const event = new Event('change', { bubbles: true });
            inputFile.dispatchEvent(event);
        }
        catch (e) {
            wrapper.invokeMethodAsync('DropAlert', e);
        }
    }, false);

2.在文件Pages/Index.razor,添加上傳組件

頁面

@implements IAsyncDisposable
@inject IJSRuntime JS

<div @ref="UploadElement" style="padding: 20px; width: 200px; height: 200px; background-color: cornflowerblue; border: 2px dashed #0087F7; border-radius: 5px; ">
    <p>拖放上傳文件</p>
     <InputFile OnChange="OnChange" class="form-control" multiple @ref="inputFile"/>
</div>

<pre>
<code>
        @uploadstatus
</code>
</pre>

代碼

  1. InputFile 開啟 multiple ,接受多文件上傳
  2. 使用JS隔離技術
  3. Dispose處理回收

@code{
    [Inject] protected Microsoft.AspNetCore.Hosting.IWebHostEnvironment? HostEnvironment { get; set; } //獲取IWebHostEnvironment

    protected ElementReference UploadElement { get; set; }
    protected InputFile? inputFile { get; set; }

    private DotNetObjectReference<Index>? wrapper;

    private IJSObjectReference? module;
    private IJSObjectReference? dropInstance;

    protected string UploadPath = "";
    protected string? uploadstatus;
    long maxFileSize = 1024 * 1024 * 15;

    protected override void OnAfterRender(bool firstRender)
    {
        if (!firstRender) return;
        UploadPath = Path.Combine(HostEnvironment!.WebRootPath, "Upload"); //初始化上傳路徑
        if (!Directory.Exists(UploadPath)) Directory.CreateDirectory(UploadPath); //不存在則新建目錄
    }

    protected async Task OnChange(InputFileChangeEventArgs e)
    {
        int i = 0;
        var selectedFiles = e.GetMultipleFiles(100);
        foreach (var item in selectedFiles)
        {
            i++;
            await OnSubmit(item);
            uploadstatus += Environment.NewLine + $"[{i}]: " + item.Name;
        }
    }

    protected async Task OnSubmit(IBrowserFile efile)
    {
        if (efile == null) return;
        var tempfilename = Path.Combine(UploadPath, efile.Name);
        await using FileStream fs = new(tempfilename, FileMode.Create);
        using var stream = efile.OpenReadStream(maxFileSize);
        await stream.CopyToAsync(fs);
        StateHasChanged();
    }


    protected override async Task OnAfterRenderAsync(bool firstRender)
    {
        if (!firstRender) return;

        module = await JS.InvokeAsync<IJSObjectReference>("import", "./lib/drag/app.js");
        wrapper = DotNetObjectReference.Create(this);
        dropInstance = await module.InvokeAsync<IJSObjectReference>("init", wrapper , UploadElement, inputFile!.Element);
    }

    [JSInvokable]
    public void DropAlert(string msg)
    {
        uploadstatus  += Environment.NewLine + $"[!Alert!]: " + msg;
        StateHasChanged();
    }


    async ValueTask IAsyncDisposable.DisposeAsync()
    {
        if (dropInstance != null)
        {
            await dropInstance.InvokeVoidAsync("dispose");
            await dropInstance.DisposeAsync();
        }

        if (wrapper != null)
        {
            wrapper.Dispose();
        }

        if (module != null)
        {
            await module.DisposeAsync();
        }
    }

}

3.就這麼簡單嗎?我們還可以加上一些騷功能

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 其中的過程適用於靜態網頁(豆瓣電影信息、嗶哩嗶哩評論區等)、動態頁面(百度圖片滾輪觸發頁面更新、下拉框觸發頁面更 新等url不變但通過滑鼠互動,致使信息更新等場景)的信息爬取。 基本適用於所有網頁信息的爬取,但代碼不夠簡潔,下述流程不夠詳細。 1 Selenium安裝 Python學習交流Q群 ...
  • 前言 又到每日分享Python小技巧的時候了,大家想看啥呢?想看的可以留言噢。今天給大家分享一個考試系統測試腳本吧,源碼對於我本人比較懶,截圖給大家,喜歡的點贊。 T007 Web-python+selenium-自動化測試技術-線上考試系統測試腳本: 正文 代碼就給大家截圖放在下麵了,需要的小伙伴 ...
  • TCP協議發送數據 A:創建Socket對象 利用Socket對象 B:獲取輸出流,寫數據 C:釋放資源 package Day26; import java.io.IOException; import java.io.OutputStream; import java.net.Socket; / ...
  • 類從被載入到虛擬機記憶體中開始,到卸載出記憶體截止,整個生命周期包括:載入、驗證、準備、解析,初始化、使用、卸載七個階段。其中驗證、準備、解析三個部分統稱為連接。 類初始化情況: 遇到new、getstatic、putstatic 或 invokestatic 這4條位元組碼指令時,如果沒有初始化,則需要 ...
  • URI在網路請求中必不可少,Spring提供了一些工具類用於解析或者生成URL,比如根據參數生成GET的URL等。本文會對Spring MVC中的URI工具進行介紹,本文主要參考Spring官方文檔。 工具類UriComponents UriComponentsBuilder可以用於根據URL和參數 ...
  • 流水淡,碧天長,鴻雁成行。編碼風格,簡捷清爽,反引無限風光。 在美劇《矽谷》中有這樣一個經典鏡頭,主人公 Richard 與同為開發工程師的女友鬧分手,理由是兩人對縮進方式有著截然不同的編程習慣,互相鄙視對方的代碼風格。Richard 認為" one tab saves four spaces ”, ...
  • @(文章目錄) 二、Java之萬年曆 2.1 要求 輸入年份; 輸入月份; 輸出某年某月的日曆。 2.2 思路 實現從控制台接收年和月,判斷是否是閏年(判斷是否是閏年:能被4整除但不能被100整除;或者能被400整除); 計算輸入月份的天數; 計算該月第一天是星期幾; 3.1 計算輸入年份距離190 ...
  • #include<iostream>using namespace std;int main(){ cout << "Hello World" << endl; system("pause"); return 0;} 在控制臺中 輸出 Hello world 變數的作用: 給一段指定的記憶體空間起名, ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...