Cara menarik lalu melepas file

Antarmuka Tarik lalu Lepas HTML memungkinkan aplikasi web menerima file yang ditarik lalu dilepas di halaman web. Selama operasi tarik lalu lepas, item file dan direktori yang ditarik akan dikaitkan dengan entri file dan entri direktori. Dalam hal menarik lalu melepas file ke browser, ada dua cara untuk melakukannya: cara modern dan klasik.

Cara modern

Menggunakan metode DataTransferItem.getAsFileSystemHandle() File System Access API

Metode DataTransferItem.getAsFileSystemHandle() akan menampilkan promise dengan objek FileSystemFileHandle jika item yang ditarik adalah file, dan promise dengan objek FileSystemDirectoryHandle jika item yang ditarik adalah direktori. Handel ini memungkinkan Anda membaca, dan secara opsional menulis kembali ke file atau direktori. Perhatikan bahwa DataTransferItem.kind antarmuka Tarik lalu Lepas akan menjadi "file" untuk file dan file, sedangkan FileSystemHandle.kind File System Access API akan menjadi "file" untuk file dan "directory" untuk direktori.

Dukungan Browser

  • 86
  • 86
  • x
  • x

Sumber

Cara klasik

Menggunakan metode DataTransferItem.getAsFile() klasik

Metode DataTransferItem.getAsFile() menampilkan objek File item data tarik. Jika item bukan file, metode ini akan menampilkan null. Meskipun Anda dapat membaca file tersebut, tidak ada cara untuk membalasnya. Metode ini memiliki kekurangan, yaitu tidak mendukung direktori.

Dukungan Browser

  • 11
  • 12
  • 50
  • 5.1

Sumber

{i>Progressive enhancement <i}

Cuplikan di bawah ini menggunakan metode DataTransferItem.getAsFileSystemHandle() File System Access API modern jika didukung, lalu melakukan fallback ke metode DataTransferItem.webkitGetAsEntry() non-standar, dan akhirnya kembali ke metode DataTransferItem.getAsFile() klasik. Pastikan untuk memeriksa jenis setiap handle, karena bisa berupa salah satu dari:

  • FileSystemFileHandle saat jalur kode modern dipilih.
  • File saat jalur kode klasik dipilih.

Semua jenis memiliki properti name, jadi logging bukanlah masalah dan akan selalu berfungsi.

// Run feature detection.
const supportsFileSystemAccessAPI =
  'getAsFileSystemHandle' in DataTransferItem.prototype;

// This is the drag and drop zone.
const elem = document.querySelector('main');

  // Prevent navigation.
elem.addEventListener('dragover', (e) => {
  e.preventDefault();
});

// Visually highlight the drop zone.
elem.addEventListener('dragenter', (e) => {
  elem.style.outline = 'solid red 1px';
});

// Visually unhighlight the drop zone.
elem.addEventListener('dragleave', (e) => {
  elem.style.outline = '';
});

// This is where the drop is handled.
elem.addEventListener('drop', async (e) => {
  // Prevent navigation.
  e.preventDefault();
  // Unhighlight the drop zone.
  elem.style.outline = '';
  // Prepare an array of promises…
  const fileHandlesPromises = [...e.dataTransfer.items]
    // …by including only files (where file misleadingly means actual file _or_
    // directory)…
    .filter((item) => item.kind === 'file')
    // …and, depending on previous feature detection…
    .map((item) =>
      supportsFileSystemAccessAPI
        // …either get a modern `FileSystemHandle`…
        ? item.getAsFileSystemHandle()
        // …or a classic `File`.
        : item.getAsFile(),
    );
  // Loop over the array of promises.
  for await (const handle of fileHandlesPromises) {
    // This is where we can actually exclusively act on the files.
    if (handle.kind === 'file' || handle.isFile) {
      console.log(`File: ${handle.name}`);
    }
  }
});

Bacaan lebih lanjut

Demo

HTML

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="utf-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1" />
    <title>How to drag and drop files</title>
  </head>
  <body>
    <main>
      <h1>How to drag and drop files</h1>
      <p>Drag and drop one or multiple files onto the page.</p>
      <pre></pre>
    </main>
  </body>
</html>

CSS


        :root {
  color-scheme: dark light;
  box-sizing: border-box;
}

*,
*:before,
*:after {
  box-sizing: inherit;
}

body {
  margin: 0;
  padding: 1rem;
  font-family: system-ui, sans-serif;
  line-height: 1.5;
  min-height: 100vh;
  display: flex;
  flex-direction: column;
}

img,
video {
  height: auto;
  max-width: 100%;
}

main {
  flex-grow: 1;
}

footer {
  margin-top: 1rem;
  border-top: solid CanvasText 1px;
  font-size: 0.8rem;
}
        

JS


        const supportsFileSystemAccessAPI =
  "getAsFileSystemHandle" in DataTransferItem.prototype;
const supportsWebkitGetAsEntry =
  "webkitGetAsEntry" in DataTransferItem.prototype;

const elem = document.querySelector("main");
const debug = document.querySelector("pre");

elem.addEventListener("dragover", (e) => {
  // Prevent navigation.
  e.preventDefault();
});

elem.addEventListener("dragenter", (e) => {
  elem.style.outline = "solid red 1px";
});

elem.addEventListener("dragleave", (e) => {
  elem.style.outline = "";
});

elem.addEventListener("drop", async (e) => {
  e.preventDefault();
  elem.style.outline = "";
  const fileHandlesPromises = [...e.dataTransfer.items]
    .filter((item) => item.kind === "file")
    .map((item) =>
      supportsFileSystemAccessAPI
        ? item.getAsFileSystemHandle()
        : supportsWebkitGetAsEntry
        ? item.webkitGetAsEntry()
        : item.getAsFile()
    );

  for await (const handle of fileHandlesPromises) {
    if (handle.kind === "directory" || handle.isDirectory) {
      console.log(`Directory: ${handle.name}`);
      debug.textContent += `Directory: ${handle.name}\n`;
    } else {
      console.log(`File: ${handle.name}`);
      debug.textContent += `File: ${handle.name}\n`;
    }
  }
});