Skip to content

Bits of .NET

Daily micro-tips for C#, SQL, performance, and scalable backend engineering.

  • Asp.Net Core
  • C#
  • SQL
  • JavaScript
  • CSS
  • About
  • ErcanOPAK.com
  • No Access
  • Privacy Policy
Ajax

Ajax: Use FormData API for File Uploads Without Libraries

- 28.05.26 - ErcanOPAK

📁 Drag & Drop + Progress Bar, Native

No more FormData polyfills. FormData API sends forms + files with fetch. Track upload progress, handle multiple files.

📝 Basic File Upload

<input type="file" id="fileInput" multiple>
<button onclick="uploadFiles()">Upload</button>
<progress id="progressBar" value="0" max="100"></progress>

async function uploadFiles() {
  const files = document.getElementById('fileInput').files;
  const formData = new FormData();
  
  for (let file of files) {
    formData.append('files[]', file);
  }
  
  const response = await fetch('/upload', {
    method: 'POST',
    body: formData
    // Don't set Content-Type — browser sets boundary automatically
  });
  
  const result = await response.json();
  console.log('Uploaded:', result);
}

🎯 Upload Progress Tracking

async function uploadWithProgress(file) {
  const formData = new FormData();
  formData.append('file', file);
  
  // Use XMLHttpRequest for progress events (Fetch doesn't support yet)
  return new Promise((resolve, reject) => {
    const xhr = new XMLHttpRequest();
    
    xhr.upload.addEventListener('progress', (e) => {
      if (e.lengthComputable) {
        const percent = (e.loaded / e.total) * 100;
        updateProgressBar(percent);
      }
    });
    
    xhr.onload = () => {
      if (xhr.status === 200) {
        resolve(JSON.parse(xhr.response));
      } else {
        reject(new Error('Upload failed'));
      }
    };
    
    xhr.open('POST', '/upload');
    xhr.send(formData);
  });
}

✅ Drag & Drop Upload

const dropZone = document.getElementById('dropZone');

dropZone.addEventListener('dragover', (e) => {
  e.preventDefault();
  dropZone.classList.add('drag-over');
});

dropZone.addEventListener('dragleave', () => {
  dropZone.classList.remove('drag-over');
});

dropZone.addEventListener('drop', async (e) => {
  e.preventDefault();
  dropZone.classList.remove('drag-over');
  
  const files = Array.from(e.dataTransfer.files);
  const formData = new FormData();
  files.forEach(file => formData.append('files[]', file));
  
  await fetch('/upload', { method: 'POST', body: formData });
});

💡 Tips

  • Don’t set Content-Type header — browser sets multipart boundary
  • Use formData.append('field', value, filename) for custom filenames
  • Access files in Node/Express: req.files (multer middleware)
  • Limit file size and type on both client and server

“Removed 50KB upload library. Built custom drag-drop upload with progress in 50 lines. FormData API is all you need for 95% of use cases.”

— Full-Stack Developer

Related posts:

Ajax: Use Axios Interceptors for Global Error Handling and Auth

Ajax: Ensuring Data Delivery with the 'keepalive' Flag

Silent JSON Parsing Failures

Post Views: 8

Post navigation

JavaScript: Use Top-Level Await in Modules for Cleaner Async Code
Git: Use Reflog to Recover Deleted Commits and Branches

Leave a Reply Cancel reply

Your email address will not be published. Required fields are marked *

June 2026
M T W T F S S
1234567
891011121314
15161718192021
22232425262728
2930  
« May    

Most Viewed Posts

  • Get the User Name and Domain Name from an Email Address in SQL (953)
  • How to add default value for Entity Framework migrations for DateTime and Bool (882)
  • Get the First and Last Word from a String or Sentence in SQL (838)
  • How to select distinct rows in a datatable in C# (808)
  • How to make theater mode the default for Youtube (805)
  • Add Constraint to SQL Table to ensure email contains @ (580)
  • How to enable, disable and check if Service Broker is enabled on a database in SQL Server (579)
  • Average of all values in a column that are not zero in SQL (538)
  • How to use Map Mode for Vertical Scroll Mode in Visual Studio (505)
  • Find numbers with more than two decimal places in SQL (454)

Recent Posts

  • C#: Use String Interpolation Instead of Concatenation
  • C#: Use Tuples to Return Multiple Values from Methods
  • SQL: Use ISNULL and NULLIF for Smart NULL Handling
  • .NET Core: Use Data Annotations for Model Validation
  • Git: Use Git Clean to Remove Untracked Files
  • Ajax: Add Custom Headers to Fetch Requests
  • JavaScript: Use console.table to Display Arrays as Tables
  • HTML: Use Spellcheck Attribute to Enable Browser Spell Check
  • CSS: Use user-select to Prevent Text Selection
  • Windows 11: Use Snipping Tool for Instant Screenshots

Most Viewed Posts

  • Get the User Name and Domain Name from an Email Address in SQL (953)
  • How to add default value for Entity Framework migrations for DateTime and Bool (882)
  • Get the First and Last Word from a String or Sentence in SQL (838)
  • How to select distinct rows in a datatable in C# (808)
  • How to make theater mode the default for Youtube (805)

Recent Posts

  • C#: Use String Interpolation Instead of Concatenation
  • C#: Use Tuples to Return Multiple Values from Methods
  • SQL: Use ISNULL and NULLIF for Smart NULL Handling
  • .NET Core: Use Data Annotations for Model Validation
  • Git: Use Git Clean to Remove Untracked Files

Social

  • ErcanOPAK.com
  • GoodReads
  • LetterBoxD
  • Linkedin
  • The Blog
  • Twitter
© 2026 Bits of .NET | Built with Xblog Plus free WordPress theme by wpthemespace.com