Added custom params, batch uploads, and progess bars
This commit is contained in:
		
							parent
							
								
									b37592e57e
								
							
						
					
					
						commit
						56b6e80e3e
					
				
							
								
								
									
										
											BIN
										
									
								
								assets/preview.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								assets/preview.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 1.5 MiB | 
							
								
								
									
										32
									
								
								index.html
									
									
									
									
									
								
							
							
						
						
									
										32
									
								
								index.html
									
									
									
									
									
								
							| @ -8,20 +8,38 @@ | ||||
| </head> | ||||
| <body> | ||||
| <div class="container"> | ||||
|     <div id="progress-container"></div> | ||||
|     <div id="error-message" style="color: red; text-align: center; display: none;">You can only upload up to 10 images.</div> | ||||
|     <h1>Instagram Formatter</h1> | ||||
|     <form id="uploadForm"> | ||||
|         <input type="file" id="imageInput" name="image" accept="image/*" required> | ||||
|         <input type="file" id="imageInput" name="image" accept="image/*" multiple required> | ||||
|         <button type="submit">Upload and Format</button> | ||||
|     </form> | ||||
|     <div class="card"> | ||||
|         <a id="card" href="#"></a> | ||||
| 
 | ||||
|     <div id="done-imgs" class="card"> | ||||
|         <a class="card-link" id="card" href="#"> | ||||
|             <img class="card-img" id="cardImg" alt="Formatted Image Preview" src="assets/preview.png"/> | ||||
|         </a> | ||||
|     </div> | ||||
| 
 | ||||
|     <!-- | ||||
|     <a id="card" style="display: none;">Download Formatted Image</a> | ||||
|     <div id="download-all-container" style="text-align: center; margin-top: 20px;"> | ||||
|         <button id="download-all" style="display: none;">Download All</button> | ||||
|     </div> | ||||
| 
 | ||||
|     <div id="preview"></div> | ||||
|     --> | ||||
|     <button type="button" class="collapsible">Advanced </button> | ||||
|     <div class="content"> | ||||
|         <label for="color">Color:</label> | ||||
|         <input type="color" id="color" name="color" placeholder="Enter color" value="#ffffff"> | ||||
| 
 | ||||
|         <label for="width">Width:</label> | ||||
|         <input type="text" id="width" name="width" placeholder="Enter width" value="1080"> | ||||
| 
 | ||||
|         <label for="height">Height:</label> | ||||
|         <input type="text" id="height" name="height" placeholder="Enter height" value="1080"> | ||||
| 
 | ||||
|         <label for="min-border">Minimum Border:</label> | ||||
|         <input type="text" id="min-border" name="min-border" placeholder="Enter minimum border" value="16"> | ||||
|     </div> | ||||
| </div> | ||||
| <script src="script.js"></script> | ||||
| </body> | ||||
|  | ||||
							
								
								
									
										181
									
								
								script.js
									
									
									
									
									
								
							
							
						
						
									
										181
									
								
								script.js
									
									
									
									
									
								
							| @ -1,42 +1,159 @@ | ||||
| document.getElementById('uploadForm').addEventListener('submit', async function (event) { | ||||
|     event.preventDefault(); | ||||
| document.addEventListener('DOMContentLoaded', function () { | ||||
|     const uploadForm = document.getElementById('uploadForm'); | ||||
|     const imageInput = document.getElementById('imageInput'); | ||||
|     const doneImgsDiv = document.getElementById('done-imgs'); | ||||
|     const progressContainer = document.getElementById('progress-container'); | ||||
|     const errorMessage = document.getElementById('error-message'); | ||||
| 
 | ||||
|     const formData = new FormData(); | ||||
|     const imageInput = document.getElementById('imageInput').files[0]; | ||||
|     // Advanced Tab Toggle Functionality
 | ||||
|     document.querySelector(".collapsible").addEventListener("click", function () { | ||||
|         this.classList.toggle("active"); | ||||
|         let content = this.nextElementSibling; | ||||
|         if (content.style.display === "block") { | ||||
|             content.style.display = "none"; | ||||
|         } else { | ||||
|             content.style.display = "block"; | ||||
|         } | ||||
|     }); | ||||
| 
 | ||||
|     if (!imageInput) { | ||||
|         alert("Please select an image to upload."); | ||||
|         return; | ||||
|     } | ||||
|     uploadForm.addEventListener('submit', function (event) { | ||||
|         document.getElementById('download-all').style.display = 'inline-block'; | ||||
|         event.preventDefault(); | ||||
| 
 | ||||
|     formData.append('image', imageInput); | ||||
|         const files = imageInput.files; | ||||
| 
 | ||||
|     try { | ||||
|         const response = await fetch('https://api.nevets.tech/igformatter/upload', { | ||||
|             method: 'POST', | ||||
|             body: formData | ||||
|         }); | ||||
| 
 | ||||
|         if (!response.ok) { | ||||
|             throw new Error('Image upload failed'); | ||||
|         if (files.length === 0) { | ||||
|             alert("Please select at least one image to upload."); | ||||
|             return; | ||||
|         } else if (files.length > 10) { | ||||
|             errorMessage.style.display = 'block'; | ||||
|             return; | ||||
|         } else { | ||||
|             errorMessage.style.display = 'none'; | ||||
|         } | ||||
| 
 | ||||
|         const arrayBuffer = await response.arrayBuffer(); | ||||
|         downloadImage(arrayBuffer, imageInput.value); | ||||
|         // Clear previous progress bars and images if needed
 | ||||
|         progressContainer.innerHTML = ''; | ||||
|         doneImgsDiv.innerHTML = ''; | ||||
| 
 | ||||
|     } catch (error) { | ||||
|         console.error('Error:', error); | ||||
|         alert('There was an error processing your image.'); | ||||
|         // Get user inputs
 | ||||
|         const color = document.getElementById("color")?.value || "#ffffff"; | ||||
|         const width = document.getElementById("width")?.value || "1080"; | ||||
|         const height = document.getElementById("height")?.value || "1080"; | ||||
|         const minBorder = document.getElementById("min-border")?.value || "16"; | ||||
| 
 | ||||
|         Array.from(files).forEach((file) => { | ||||
|             uploadAndProcessImage(file, { color, width, height, minBorder }); | ||||
|         }); | ||||
|     }); | ||||
| 
 | ||||
|     function uploadAndProcessImage(file, options) { | ||||
|         const { color, width, height, minBorder } = options; | ||||
| 
 | ||||
|         // Create progress bar elements
 | ||||
|         const progressWrapper = document.createElement('div'); | ||||
|         progressWrapper.className = 'progress-wrapper'; | ||||
| 
 | ||||
|         const fileNameLabel = document.createElement('div'); | ||||
|         fileNameLabel.className = 'file-name'; | ||||
|         fileNameLabel.textContent = file.name; | ||||
| 
 | ||||
|         const progressBar = document.createElement('div'); | ||||
|         progressBar.className = 'progress-bar'; | ||||
| 
 | ||||
|         const progressBarFill = document.createElement('div'); | ||||
|         progressBarFill.className = 'progress-bar-fill'; | ||||
| 
 | ||||
|         const progressStatus = document.createElement('div'); | ||||
|         progressStatus.className = 'progress-status'; | ||||
|         progressStatus.textContent = 'Starting upload...'; | ||||
| 
 | ||||
|         progressBar.appendChild(progressBarFill); | ||||
|         progressWrapper.appendChild(fileNameLabel); | ||||
|         progressWrapper.appendChild(progressBar); | ||||
|         progressWrapper.appendChild(progressStatus); | ||||
|         progressContainer.appendChild(progressWrapper); | ||||
| 
 | ||||
|         const formData = new FormData(); | ||||
|         formData.append('image', file); | ||||
| 
 | ||||
|         const xhr = new XMLHttpRequest(); | ||||
| 
 | ||||
|         xhr.open('POST', `https://api.nevets.tech/igformatter/upload?color=${encodeURIComponent(color)}&width=${encodeURIComponent(width)}&height=${encodeURIComponent(height)}&minBorder=${encodeURIComponent(minBorder)}`, true); | ||||
|         xhr.responseType = 'blob'; | ||||
| 
 | ||||
|         // Upload progress event
 | ||||
|         xhr.upload.addEventListener('progress', function (e) { | ||||
|             if (e.lengthComputable) { | ||||
|                 const percentComplete = ((e.loaded / e.total) * 50).toFixed(2); // 0% to 50%
 | ||||
|                 progressBarFill.style.width = `${percentComplete}%`; | ||||
|                 progressStatus.textContent = `Uploading... ${percentComplete}%`; | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|         // Download progress event
 | ||||
|         xhr.addEventListener('progress', function (e) { | ||||
|             if (e.lengthComputable) { | ||||
|                 const percentComplete = (50 + ((e.loaded / e.total) * 50)).toFixed(2); // 50% to 100%
 | ||||
|                 progressBarFill.style.width = `${percentComplete}%`; | ||||
|                 progressStatus.textContent = `Processing... ${percentComplete}%`; | ||||
|             } else { | ||||
|                 progressStatus.textContent = `Processing...`; | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|         // On successful completion
 | ||||
|         xhr.addEventListener('load', function () { | ||||
|             if (xhr.status === 200) { | ||||
|                 progressBarFill.style.width = '100%'; | ||||
|                 progressStatus.textContent = 'Completed'; | ||||
|                 progressWrapper.classList.add('success'); | ||||
| 
 | ||||
|                 const blob = xhr.response; | ||||
|                 const imageUrl = URL.createObjectURL(blob); | ||||
| 
 | ||||
|                 const anchor = document.createElement('a'); | ||||
|                 anchor.href = imageUrl; | ||||
|                 anchor.className = "card-link"; | ||||
|                 anchor.download = `${file.name.replace(/\.[^/.]+$/, "")}_formatted.png`; | ||||
| 
 | ||||
|                 const img = document.createElement('img'); | ||||
|                 img.src = imageUrl; | ||||
|                 img.className = "card-img"; | ||||
|                 img.alt = "Formatted Image Preview"; | ||||
| 
 | ||||
|                 anchor.appendChild(img); | ||||
|                 doneImgsDiv.appendChild(anchor); | ||||
|             } else { | ||||
|                 handleError(`Processing failed for ${file.name}`, progressWrapper, progressStatus, progressBarFill); | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|         // On error
 | ||||
|         xhr.addEventListener('error', function () { | ||||
|             handleError(`An error occurred while uploading ${file.name}`, progressWrapper, progressStatus, progressBarFill); | ||||
|         }); | ||||
| 
 | ||||
|         // On abort
 | ||||
|         xhr.addEventListener('abort', function () { | ||||
|             handleError(`Upload aborted for ${file.name}`, progressWrapper, progressStatus, progressBarFill); | ||||
|         }); | ||||
| 
 | ||||
|         xhr.send(formData); | ||||
|     } | ||||
| 
 | ||||
|     function handleError(message, progressWrapper, progressStatus, progressBarFill) { | ||||
|         console.error(message); | ||||
|         progressStatus.textContent = message; | ||||
|         progressWrapper.classList.add('error'); | ||||
|         progressBarFill.style.width = '100%'; | ||||
|     } | ||||
| }); | ||||
| 
 | ||||
| function downloadImage(arrayBuffer, fileName) { | ||||
|     const blob = new Blob([arrayBuffer], { type: "image/png" }); | ||||
|     const cardElement = document.getElementById("card"); | ||||
| 
 | ||||
|     let urlCreator = window.URL || window.webkitURL; | ||||
|     cardElement.href = urlCreator.createObjectURL(blob); | ||||
|     cardElement.style.display = 'unset'; | ||||
|     cardElement.download = fileName + ".png"; | ||||
|     cardElement.click(); | ||||
| } | ||||
| // Handle the download all functionality
 | ||||
| document.getElementById('download-all').addEventListener('click', function() { | ||||
|     const links = document.querySelectorAll('#done-imgs .card-link'); | ||||
|     links.forEach(link => { | ||||
|         link.click(); | ||||
|     }); | ||||
| }); | ||||
|  | ||||
							
								
								
									
										169
									
								
								styles.css
									
									
									
									
									
								
							
							
						
						
									
										169
									
								
								styles.css
									
									
									
									
									
								
							| @ -3,7 +3,7 @@ body { | ||||
|     background-color: #f4f4f4; | ||||
|     display: flex; | ||||
|     justify-content: center; | ||||
|     align-items: center; | ||||
|     align-items: flex-start; | ||||
|     height: 100vh; | ||||
|     margin: 0; | ||||
| } | ||||
| @ -12,8 +12,10 @@ body { | ||||
|     background: white; | ||||
|     padding: 20px; | ||||
|     border-radius: 10px; | ||||
|     box-shadow: 0px 0px 20px rgba(0, 0, 0, 0.1); | ||||
|     box-shadow: 0 0 20px rgba(0, 0, 0, 0.1); | ||||
|     text-align: center; | ||||
|     max-width: 800px; | ||||
|     margin: auto; | ||||
| } | ||||
| 
 | ||||
| h1 { | ||||
| @ -23,6 +25,9 @@ h1 { | ||||
| 
 | ||||
| input[type="file"] { | ||||
|     margin-bottom: 15px; | ||||
|     display: block; | ||||
|     margin-left: auto; | ||||
|     margin-right: auto; | ||||
| } | ||||
| 
 | ||||
| button { | ||||
| @ -38,20 +43,160 @@ button:hover { | ||||
|     background-color: #0a85d9; | ||||
| } | ||||
| 
 | ||||
| #card { | ||||
|     margin-top: 20px; | ||||
|     display: inline-block; | ||||
|     padding: 10px 20px; | ||||
|     background-color: #28a745; | ||||
|     color: white; | ||||
| /* Center the progress bars */ | ||||
| #progress-container { | ||||
|     width: 100%; | ||||
|     max-width: 800px; | ||||
|     margin: 20px auto; | ||||
|     text-align: center; | ||||
| } | ||||
| 
 | ||||
| .progress-wrapper { | ||||
|     margin-bottom: 10px; | ||||
|     padding: 10px; | ||||
|     background-color: #f9f9f9; | ||||
|     border: 1px solid #ddd; | ||||
|     border-radius: 5px; | ||||
|     text-decoration: none; | ||||
|     width: 100%; | ||||
|     max-width: 400px; | ||||
|     margin-left: auto; | ||||
|     margin-right: auto; | ||||
| } | ||||
| 
 | ||||
| #card:hover { | ||||
|     background-color: #218838; | ||||
| .progress-wrapper .file-name { | ||||
|     margin-bottom: 5px; | ||||
|     font-weight: bold; | ||||
|     color: #333; | ||||
| } | ||||
| 
 | ||||
| #preview { | ||||
| .progress-bar { | ||||
|     width: 100%; | ||||
|     background-color: #e6e6e6; | ||||
|     border-radius: 5px; | ||||
|     overflow: hidden; | ||||
|     height: 20px; | ||||
|     position: relative; | ||||
| } | ||||
| 
 | ||||
| .progress-bar-fill { | ||||
|     height: 100%; | ||||
|     width: 0%; | ||||
|     background-color: #1da1f2; | ||||
|     transition: width 0.2s ease; | ||||
| } | ||||
| 
 | ||||
| .progress-status { | ||||
|     margin-top: 5px; | ||||
|     font-size: 12px; | ||||
|     color: #555; | ||||
| } | ||||
| 
 | ||||
| .progress-wrapper.success .progress-bar-fill { | ||||
|     background-color: #28a745; | ||||
| } | ||||
| 
 | ||||
| .progress-wrapper.success .progress-status { | ||||
|     color: #28a745; | ||||
| } | ||||
| 
 | ||||
| .progress-wrapper.error .progress-bar-fill { | ||||
|     background-color: #dc3545; | ||||
| } | ||||
| 
 | ||||
| .progress-wrapper.error .progress-status { | ||||
|     color: #dc3545; | ||||
| } | ||||
| 
 | ||||
| /* Responsive grid for images */ | ||||
| #done-imgs { | ||||
|     display: flex; | ||||
|     flex-wrap: wrap; | ||||
|     justify-content: center; | ||||
|     margin-top: 20px; | ||||
| } | ||||
| 
 | ||||
| .card-link { | ||||
|     flex: 1 1 calc(25% - 20px); /* 4 images per row with space */ | ||||
|     margin: 10px; | ||||
|     max-width: calc(25% - 20px); | ||||
|     text-align: center; | ||||
| } | ||||
| 
 | ||||
| .card-link img { | ||||
|     width: 100%; | ||||
|     height: auto; | ||||
|     border: 2px solid gray; | ||||
|     border-radius: 5px; | ||||
| } | ||||
| 
 | ||||
| /* Advanced tab styling */ | ||||
| .collapsible { | ||||
|     background-color: #1da1f2; | ||||
|     color: white; | ||||
|     padding: 10px 20px; | ||||
|     border: none; | ||||
|     border-radius: 5px; | ||||
|     cursor: pointer; | ||||
|     width: 100%; | ||||
|     text-align: center; | ||||
|     outline: none; | ||||
|     font-size: 16px; | ||||
|     margin-top: 20px; | ||||
| } | ||||
| 
 | ||||
| .collapsible:hover { | ||||
|     background-color: #0a85d9; | ||||
| } | ||||
| 
 | ||||
| .collapsible:after { | ||||
|     content: '\002B'; /* Unicode character for "plus" sign (+) */ | ||||
|     font-weight: bold; | ||||
| } | ||||
| 
 | ||||
| .collapsible.active:after { | ||||
|     content: "\2212"; /* Unicode character for "minus" sign (−) */ | ||||
| } | ||||
| 
 | ||||
| .content { | ||||
|     padding: 10px 20px; | ||||
|     display: none; | ||||
|     overflow: hidden; | ||||
|     background-color: #f1f1f1; | ||||
|     margin-top: 10px; | ||||
|     border-radius: 5px; | ||||
|     text-align: left; | ||||
| } | ||||
| 
 | ||||
| .content input[type="text"], | ||||
| .content input[type="number"], | ||||
| .content input[type="color"] { | ||||
|     margin: 5px 10px; | ||||
|     padding: 8px; | ||||
|     width: 80px; | ||||
|     box-sizing: border-box; | ||||
|     border: 1px solid #ccc; | ||||
|     border-radius: 5px; | ||||
|     display: inline-block; | ||||
|     vertical-align: middle; | ||||
| } | ||||
| 
 | ||||
| .content label { | ||||
|     margin: 5px 10px; | ||||
|     display: inline-block; | ||||
|     text-align: left; | ||||
|     color: #333; | ||||
|     vertical-align: middle; | ||||
| } | ||||
| 
 | ||||
| .content input[type="color"] { | ||||
|     width: 50px; | ||||
|     height: 30px; | ||||
|     padding: 0; | ||||
|     border: 1px solid #ccc; | ||||
|     border-radius: 5px; | ||||
|     cursor: pointer; | ||||
|     background-color: #fff; | ||||
|     -webkit-appearance: none; | ||||
|     appearance: none; | ||||
|     box-shadow: inset 0 1px 3px rgba(0, 0, 0, 0.1); | ||||
| } | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user