mirror of
				https://github.com/C4illin/ConvertX.git
				synced 2025-10-23 04:52:18 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			252 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			252 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| const webroot = document.querySelector("meta[name='webroot']").content;
 | |
| const fileInput = document.querySelector('input[type="file"]');
 | |
| const dropZone = document.getElementById("dropzone");
 | |
| const convertButton = document.querySelector("input[type='submit']");
 | |
| const fileNames = [];
 | |
| let fileType;
 | |
| let pendingFiles = 0;
 | |
| let formatSelected = false;
 | |
| 
 | |
| dropZone.addEventListener("dragover", (e) => {
 | |
|   e.preventDefault();
 | |
|   dropZone.classList.add("dragover");
 | |
| });
 | |
| 
 | |
| dropZone.addEventListener("dragleave", () => {
 | |
|   dropZone.classList.remove("dragover");
 | |
| });
 | |
| 
 | |
| dropZone.addEventListener("drop", (e) => {
 | |
|   e.preventDefault();
 | |
|   dropZone.classList.remove("dragover");
 | |
| 
 | |
|   const files = e.dataTransfer.files;
 | |
| 
 | |
|   if (files.length === 0) {
 | |
|     console.warn("No files dropped — likely a URL or unsupported source.");
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   for (const file of files) {
 | |
|     console.log("Handling dropped file:", file.name);
 | |
|     handleFile(file);
 | |
|   }
 | |
| });
 | |
| 
 | |
| // Extracted handleFile function for reusability in drag-and-drop and file input
 | |
| function handleFile(file) {
 | |
|   const fileList = document.querySelector("#file-list");
 | |
| 
 | |
|   const row = document.createElement("tr");
 | |
|   row.innerHTML = `
 | |
|     <td>${file.name}</td>
 | |
|     <td><progress max="100" class="inline-block h-2 appearance-none overflow-hidden rounded-full border-0 bg-neutral-700 bg-none text-accent-500 accent-accent-500 [&::-moz-progress-bar]:bg-accent-500 [&::-webkit-progress-value]:rounded-full [&::-webkit-progress-value]:[background:none] [&[value]::-webkit-progress-value]:bg-accent-500 [&[value]::-webkit-progress-value]:transition-[inline-size]"></progress></td>
 | |
|     <td>${(file.size / 1024).toFixed(2)} kB</td>
 | |
|     <td><a onclick="deleteRow(this)">Remove</a></td>
 | |
|   `;
 | |
| 
 | |
|   if (!fileType) {
 | |
|     fileType = file.name.split(".").pop();
 | |
|     fileInput.setAttribute("accept", `.${fileType}`);
 | |
|     setTitle();
 | |
| 
 | |
|     fetch(`${webroot}/conversions`, {
 | |
|       method: "POST",
 | |
|       body: JSON.stringify({ fileType }),
 | |
|       headers: { "Content-Type": "application/json" },
 | |
|     })
 | |
|       .then((res) => res.text())
 | |
|       .then((html) => {
 | |
|         selectContainer.innerHTML = html;
 | |
|         updateSearchBar();
 | |
|       })
 | |
|       .catch(console.error);
 | |
|   }
 | |
| 
 | |
|   fileList.appendChild(row);
 | |
|   file.htmlRow = row;
 | |
|   fileNames.push(file.name);
 | |
|   uploadFile(file);
 | |
| }
 | |
| 
 | |
| const selectContainer = document.querySelector("form .select_container");
 | |
| 
 | |
| const updateSearchBar = () => {
 | |
|   const convertToInput = document.querySelector("input[name='convert_to_search']");
 | |
|   const convertToPopup = document.querySelector(".convert_to_popup");
 | |
|   const convertToGroupElements = document.querySelectorAll(".convert_to_group");
 | |
|   const convertToGroups = {};
 | |
|   const convertToElement = document.querySelector("select[name='convert_to']");
 | |
| 
 | |
|   const showMatching = (search) => {
 | |
|     for (const [targets, groupElement] of Object.values(convertToGroups)) {
 | |
|       let matchingTargetsFound = 0;
 | |
|       for (const target of targets) {
 | |
|         if (target.dataset.target.includes(search)) {
 | |
|           matchingTargetsFound++;
 | |
|           target.classList.remove("hidden");
 | |
|           target.classList.add("flex");
 | |
|         } else {
 | |
|           target.classList.add("hidden");
 | |
|           target.classList.remove("flex");
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (matchingTargetsFound === 0) {
 | |
|         groupElement.classList.add("hidden");
 | |
|         groupElement.classList.remove("flex");
 | |
|       } else {
 | |
|         groupElement.classList.remove("hidden");
 | |
|         groupElement.classList.add("flex");
 | |
|       }
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   for (const groupElement of convertToGroupElements) {
 | |
|     const groupName = groupElement.dataset.converter;
 | |
| 
 | |
|     const targetElements = groupElement.querySelectorAll(".target");
 | |
|     const targets = Array.from(targetElements);
 | |
| 
 | |
|     for (const target of targets) {
 | |
|       target.onmousedown = () => {
 | |
|         convertToElement.value = target.dataset.value;
 | |
|         convertToInput.value = `${target.dataset.target} using ${target.dataset.converter}`;
 | |
|         formatSelected = true;
 | |
|         if (pendingFiles === 0 && fileNames.length > 0) {
 | |
|           convertButton.disabled = false;
 | |
|         }
 | |
|         showMatching("");
 | |
|       };
 | |
|     }
 | |
| 
 | |
|     convertToGroups[groupName] = [targets, groupElement];
 | |
|   }
 | |
| 
 | |
|   convertToInput.addEventListener("input", (e) => {
 | |
|     showMatching(e.target.value.toLowerCase());
 | |
|   });
 | |
| 
 | |
|   convertToInput.addEventListener("search", () => {
 | |
|     // when the user clears the search bar using the 'x' button
 | |
|     convertButton.disabled = true;
 | |
|     formatSelected = false;
 | |
|   });
 | |
| 
 | |
|   convertToInput.addEventListener("blur", (e) => {
 | |
|     // Keep the popup open even when clicking on a target button
 | |
|     // for a split second to allow the click to go through
 | |
|     if (e?.relatedTarget?.classList?.contains("target")) {
 | |
|       convertToPopup.classList.add("hidden");
 | |
|       convertToPopup.classList.remove("flex");
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     convertToPopup.classList.add("hidden");
 | |
|     convertToPopup.classList.remove("flex");
 | |
|   });
 | |
| 
 | |
|   convertToInput.addEventListener("focus", () => {
 | |
|     convertToPopup.classList.remove("hidden");
 | |
|     convertToPopup.classList.add("flex");
 | |
|   });
 | |
| };
 | |
| 
 | |
| // Add a 'change' event listener to the file input element
 | |
| fileInput.addEventListener("change", (e) => {
 | |
|   const files = e.target.files;
 | |
|   for (const file of files) {
 | |
|     handleFile(file);
 | |
|   }
 | |
| });
 | |
| 
 | |
| const setTitle = () => {
 | |
|   const title = document.querySelector("h1");
 | |
|   title.textContent = `Convert ${fileType ? `.${fileType}` : ""}`;
 | |
| };
 | |
| 
 | |
| // Add a onclick for the delete button
 | |
| // eslint-disable-next-line @typescript-eslint/no-unused-vars
 | |
| const deleteRow = (target) => {
 | |
|   const filename = target.parentElement.parentElement.children[0].textContent;
 | |
|   const row = target.parentElement.parentElement;
 | |
|   row.remove();
 | |
| 
 | |
|   // remove from fileNames
 | |
|   const index = fileNames.indexOf(filename);
 | |
|   fileNames.splice(index, 1);
 | |
| 
 | |
|   // reset fileInput
 | |
|   fileInput.value = "";
 | |
| 
 | |
|   // if fileNames is empty, reset fileType
 | |
|   if (fileNames.length === 0) {
 | |
|     fileType = null;
 | |
|     fileInput.removeAttribute("accept");
 | |
|     convertButton.disabled = true;
 | |
|     setTitle();
 | |
|   }
 | |
| 
 | |
|   fetch(`${webroot}/delete`, {
 | |
|     method: "POST",
 | |
|     body: JSON.stringify({ filename: filename }),
 | |
|     headers: {
 | |
|       "Content-Type": "application/json",
 | |
|     },
 | |
|   }).catch((err) => console.log(err));
 | |
| };
 | |
| 
 | |
| const uploadFile = (file) => {
 | |
|   convertButton.disabled = true;
 | |
|   convertButton.textContent = "Uploading...";
 | |
|   pendingFiles += 1;
 | |
| 
 | |
|   const formData = new FormData();
 | |
|   formData.append("file", file, file.name);
 | |
| 
 | |
|   let xhr = new XMLHttpRequest();
 | |
| 
 | |
|   xhr.open("POST", `${webroot}/upload`, true);
 | |
| 
 | |
|   xhr.onload = () => {
 | |
|     let data = JSON.parse(xhr.responseText);
 | |
| 
 | |
|     pendingFiles -= 1;
 | |
|     if (pendingFiles === 0) {
 | |
|       if (formatSelected) {
 | |
|         convertButton.disabled = false;
 | |
|       }
 | |
|       convertButton.textContent = "Convert";
 | |
|     }
 | |
| 
 | |
|     //Remove the progress bar when upload is done
 | |
|     let progressbar = file.htmlRow.getElementsByTagName("progress");
 | |
|     progressbar[0].parentElement.remove();
 | |
|     console.log(data);
 | |
|   };
 | |
| 
 | |
|   xhr.upload.onprogress = (e) => {
 | |
|     let sent = e.loaded;
 | |
|     let total = e.total;
 | |
|     console.log(`upload progress (${file.name}):`, (100 * sent) / total);
 | |
| 
 | |
|     let progressbar = file.htmlRow.getElementsByTagName("progress");
 | |
|     progressbar[0].value = (100 * sent) / total;
 | |
|   };
 | |
| 
 | |
|   xhr.onerror = (e) => {
 | |
|     console.log(e);
 | |
|   };
 | |
| 
 | |
|   xhr.send(formData);
 | |
| };
 | |
| 
 | |
| const formConvert = document.querySelector(`form[action='${webroot}/convert']`);
 | |
| 
 | |
| formConvert.addEventListener("submit", () => {
 | |
|   const hiddenInput = document.querySelector("input[name='file_names']");
 | |
|   hiddenInput.value = JSON.stringify(fileNames);
 | |
| });
 | |
| 
 | |
| updateSearchBar();
 |