mirror of
https://github.com/BobbyWibowo/lolisafe.git
synced 2025-01-19 01:31:34 +00:00
a6156da798
Their thumbs can be made with ffmpeg Run "yarn thumbs 2" to create thumbs for existing uploads with those extensions Rebuilt client assets and bumped v1 version string
1 line
54 KiB
Plaintext
1 line
54 KiB
Plaintext
{"version":3,"sources":["home.js"],"names":["const","lsKeys","token","chunkSize","parallelUploads","uploadsHistoryOrder","previewImages","fileLength","uploadAge","stripTags","page","localStorage","private","enableUserAccounts","maxSize","temporaryUploadAges","fileIdentifierLength","stripTagsConfig","album","maxSizeBytes","urlMaxSize","urlMaxSizeBytes","tabs","activeTab","albumSelect","albumSelectOnChange","previewTemplate","dropzone","clipboardJS","lazyLoad","urlsQueue","activeUrlsQueue","imageExts","videoExts","albumTitleMaxLength","albumDescMaxLength","onInitError","error","document","querySelector","classList","add","querySelectorAll","forEach","element","uploadButton","innerText","remove","addEventListener","location","reload","response","onAxiosError","onError","console","content","createElement","innerHTML","toString","swal","title","icon","statusText","520","521","522","523","524","525","526","527","530","status","description","data","checkIfPublic","let","renderShown","axios","get","onDownloadProgress","doRender","then","parseInt","preparePage","catch","prepareUpload","verifyToken","button","href","reloadOnError","post","success","text","removeItem","value","prepareShareX","fetchAlbums","prepareUploadConfig","getPrettyBytes","prepareDropzone","event","addUrlsToQueue","tabsContainer","loop","i","id","dataset","tabContent","setActiveTab","push","tab","length","index","headers","Array","isArray","albums","option","name","appendChild","previewNode","parentNode","removeChild","tabDiv","div","className","previewsContainer","Dropzone","body","url","paramName","clickable","maxFilesize","uploadMultiple","createImageThumbnails","autoProcessQueue","chunking","Boolean","parallelChunkUploads","timeout","init","this","on","file","previewElement","xhr","ontimeout","instances","getUploadingFiles","filter","instance","_handleUploadError","undefined","_start","Date","now","upload","chunked","setRequestHeader","chunks","progress","prettyBytesPerSec","total","Math","max","size","percentage","bytesSent","toFixed","upl","prefix","skipProgress","done","last","totalChunkCount","chunkIndex","elapsed","bytesPerSec","files","updateTemplate","test","updateTemplateIcon","chunksUploaded","uuid","original","type","albumid","filelength","age","striptags","urls","split","trim","firstChild","processUrlsQueue","shiftQueue","initUrlUpload","match","finishedUrlUpload","shift","templateElement","iconClass","iconElement","link","a","clipboard","clipboardText","parentElement","exec","extname","toLowerCase","includes","img","setAttribute","src","onerror","currentTarget","update","expirydate","expiryDate","getPrettyDate","createAlbum","buttons","cancel","confirm","closeModal","download","checked","public","selected","fallback","min","config","siBytes","label","select","help","valueHandler","display","number","round","disabled","force","default","video","isNaN","suffix","uploadFields","stored","parseFloat","String","getPrettyUploadAge","form","preventDefault","configKeys","Object","keys","key","conf","field","parsed","find","sel","control","opts","j","opt","join","tmp","submit","checkValidity","elements","hours","minutes","days","window","items","clipboardData","originalEvent","item","kind","blob","getAsFile","File","addFile","onload","ClipboardJS","timer","LazyLoad","elements_selector"],"mappings":"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","file":"home.js","sourcesContent":["/* global swal, axios, Dropzone, ClipboardJS, LazyLoad */\n\nconst lsKeys = {\n token: 'token',\n chunkSize: 'chunkSize',\n parallelUploads: 'parallelUploads',\n uploadsHistoryOrder: 'uploadsHistoryOrder',\n previewImages: 'previewImages',\n fileLength: 'fileLength',\n uploadAge: 'uploadAge',\n stripTags: 'stripTags'\n}\n\nconst page = {\n // user token\n token: localStorage[lsKeys.token],\n\n // configs from api/check\n private: null,\n enableUserAccounts: null,\n maxSize: null,\n chunkSize: null,\n temporaryUploadAges: null,\n fileIdentifierLength: null,\n stripTagsConfig: null,\n\n // store album id that will be used with upload requests\n album: null,\n\n parallelUploads: 2,\n previewImages: null,\n fileLength: null,\n uploadAge: null,\n\n maxSizeBytes: null,\n urlMaxSize: null,\n urlMaxSizeBytes: null,\n\n tabs: [],\n activeTab: null,\n albumSelect: null,\n albumSelectOnChange: null,\n previewTemplate: null,\n\n dropzone: null,\n clipboardJS: null,\n lazyLoad: null,\n\n // additional vars for url uploads\n urlsQueue: [],\n activeUrlsQueue: 0,\n\n // Include BMP for uploads preview only, cause the real images will be used\n // Sharp isn't capable of making their thumbnails for dashboard and album public pages\n imageExts: ['.webp', '.jpg', '.jpeg', '.bmp', '.gif', '.png', '.tiff', '.tif', '.svg'],\n videoExts: ['.webm', '.mp4', '.wmv', '.avi', '.mov', '.mkv', '.m4v', '.m2ts'],\n\n albumTitleMaxLength: 70,\n albumDescMaxLength: 4000\n}\n\n// Handler for errors during initialization\npage.onInitError = error => {\n // Hide these elements\n document.querySelector('#albumDiv').classList.add('is-hidden')\n document.querySelector('#tabs').classList.add('is-hidden')\n document.querySelectorAll('.tab-content').forEach(element => {\n return element.classList.add('is-hidden')\n })\n\n // Update upload button\n const uploadButton = document.querySelector('#loginToUpload')\n uploadButton.innerText = 'An error occurred. Try to reload?'\n uploadButton.classList.remove('is-loading')\n uploadButton.classList.remove('is-hidden')\n\n uploadButton.addEventListener('click', () => {\n location.reload()\n })\n\n if (error.response)\n page.onAxiosError(error)\n else\n page.onError(error)\n}\n\n// Handler for regular JS errors\npage.onError = error => {\n console.error(error)\n\n const content = document.createElement('div')\n content.innerHTML = `<code>${error.toString()}</code>`\n return swal({\n title: 'An error occurred!',\n icon: 'error',\n content\n })\n}\n\n// Handler for Axios errors\npage.onAxiosError = error => {\n console.error(error)\n\n // Better Cloudflare errors\n const cloudflareErrors = {\n 520: 'Unknown Error',\n 521: 'Web Server Is Down',\n 522: 'Connection Timed Out',\n 523: 'Origin Is Unreachable',\n 524: 'A Timeout Occurred',\n 525: 'SSL Handshake Failed',\n 526: 'Invalid SSL Certificate',\n 527: 'Railgun Error',\n 530: 'Origin DNS Error'\n }\n\n const statusText = cloudflareErrors[error.response.status] || error.response.statusText\n const description = error.response.data && error.response.data.description\n ? error.response.data.description\n : 'There was an error with the request, please check the console for more information.'\n\n return swal(`${error.response.status} ${statusText}`, description, 'error')\n}\n\npage.checkIfPublic = () => {\n let renderShown = false\n return axios.get('api/check', {\n onDownloadProgress: () => {\n // Only show render after this request has been initiated\n if (!renderShown && typeof page.doRender === 'function') {\n page.doRender()\n renderShown = true\n }\n }\n }).then(response => {\n page.private = response.data.private\n page.enableUserAccounts = response.data.enableUserAccounts\n page.maxSize = parseInt(response.data.maxSize)\n page.maxSizeBytes = page.maxSize * 1e6\n page.chunkSize = parseInt(response.data.chunkSize)\n page.temporaryUploadAges = response.data.temporaryUploadAges\n page.fileIdentifierLength = response.data.fileIdentifierLength\n page.stripTagsConfig = response.data.stripTags\n return page.preparePage()\n }).catch(page.onInitError)\n}\n\npage.preparePage = () => {\n if (page.private)\n if (page.token) {\n return page.verifyToken(page.token, true)\n } else {\n const button = document.querySelector('#loginToUpload')\n button.href = 'auth'\n button.classList.remove('is-loading')\n if (page.enableUserAccounts)\n button.innerText = 'Anonymous upload is disabled. Log in to upload.'\n else\n button.innerText = 'Running in private mode. Log in to upload.'\n }\n else\n return page.prepareUpload()\n}\n\npage.verifyToken = (token, reloadOnError) => {\n return axios.post('api/tokens/verify', { token }).then(response => {\n if (response.data.success === false)\n return swal({\n title: 'An error occurred!',\n text: response.data.description,\n icon: 'error'\n }).then(() => {\n if (!reloadOnError) return\n localStorage.removeItem('token')\n location.reload()\n })\n\n localStorage[lsKeys.token] = token\n page.token = token\n return page.prepareUpload()\n }).catch(page.onInitError)\n}\n\npage.prepareUpload = () => {\n // I think this fits best here because we need to check for a valid token before we can get the albums\n if (page.token) {\n // Display the album selection\n document.querySelector('#albumDiv').classList.remove('is-hidden')\n\n page.albumSelect = document.querySelector('#albumSelect')\n page.albumSelectOnChange = () => {\n page.album = parseInt(page.albumSelect.value)\n // Re-generate ShareX config file\n if (typeof page.prepareShareX === 'function')\n page.prepareShareX()\n }\n page.albumSelect.addEventListener('change', page.albumSelectOnChange)\n\n // Fetch albums\n page.fetchAlbums()\n }\n\n // Prepare & generate config tab\n page.prepareUploadConfig()\n\n // Update elements wherever applicable\n document.querySelector('#maxSize > span').innerHTML = page.getPrettyBytes(page.maxSizeBytes)\n document.querySelector('#loginToUpload').classList.add('is-hidden')\n\n if (!page.token && page.enableUserAccounts)\n document.querySelector('#loginLinkText').innerHTML = 'Create an account and keep track of your uploads'\n\n // Prepare & generate files upload tab\n page.prepareDropzone()\n\n // Generate ShareX config file\n if (typeof page.prepareShareX === 'function')\n page.prepareShareX()\n\n // Prepare urls upload tab\n const urlMaxSize = document.querySelector('#urlMaxSize')\n if (urlMaxSize) {\n page.urlMaxSize = parseInt(urlMaxSize.innerHTML)\n page.urlMaxSizeBytes = page.urlMaxSize * 1e6\n urlMaxSize.innerHTML = page.getPrettyBytes(page.urlMaxSizeBytes)\n document.querySelector('#uploadUrls').addEventListener('click', event => {\n page.addUrlsToQueue()\n })\n }\n\n // Get all tabs\n const tabsContainer = document.querySelector('#tabs')\n const tabs = tabsContainer.querySelectorAll('li')\n for (let i = 0; i < tabs.length; i++) {\n const id = tabs[i].dataset.id\n const tabContent = document.querySelector(`#${id}`)\n if (!tabContent) continue\n\n tabs[i].addEventListener('click', () => {\n page.setActiveTab(i)\n })\n page.tabs.push({ tab: tabs[i], content: tabContent })\n }\n\n // Set first valid tab as the default active tab\n if (page.tabs.length) {\n page.setActiveTab(0)\n tabsContainer.classList.remove('is-hidden')\n }\n}\n\npage.setActiveTab = index => {\n for (let i = 0; i < page.tabs.length; i++)\n if (i === index) {\n page.tabs[i].tab.classList.add('is-active')\n page.tabs[i].content.classList.remove('is-hidden')\n page.activeTab = index\n } else {\n page.tabs[i].tab.classList.remove('is-active')\n page.tabs[i].content.classList.add('is-hidden')\n }\n}\n\npage.fetchAlbums = () => {\n return axios.get('api/albums', { headers: { token: page.token } }).then(response => {\n if (response.data.success === false)\n return swal('An error occurred!', response.data.description, 'error')\n\n // Create an option for each album\n if (Array.isArray(response.data.albums) && response.data.albums.length)\n for (let i = 0; i < response.data.albums.length; i++) {\n const album = response.data.albums[i]\n const option = document.createElement('option')\n option.value = album.id\n option.innerHTML = album.name\n page.albumSelect.appendChild(option)\n }\n }).catch(page.onInitError)\n}\n\npage.prepareDropzone = () => {\n // Parse template element\n const previewNode = document.querySelector('#tpl')\n page.previewTemplate = previewNode.innerHTML\n previewNode.parentNode.removeChild(previewNode)\n\n // Generate files upload tab\n const tabDiv = document.querySelector('#tab-files')\n const div = document.createElement('div')\n div.className = 'control is-expanded'\n div.innerHTML = `\n <div id=\"dropzone\" class=\"button is-danger is-outlined is-fullwidth is-unselectable\">\n <span class=\"icon\">\n <i class=\"icon-upload-cloud\"></i>\n </span>\n <span>Click here or drag & drop files</span>\n </div>\n `\n tabDiv.querySelector('.dz-container').appendChild(div)\n\n const previewsContainer = tabDiv.querySelector('#tab-files .field.uploads')\n\n page.dropzone = new Dropzone(document.body, {\n url: 'api/upload',\n paramName: 'files[]',\n clickable: tabDiv.querySelector('#dropzone'),\n maxFilesize: page.maxSizeBytes / 1024 / 1024, // this option expects MiB\n parallelUploads: page.parallelUploads,\n uploadMultiple: false,\n previewsContainer,\n previewTemplate: page.previewTemplate,\n createImageThumbnails: false,\n autoProcessQueue: true,\n headers: { token: page.token },\n chunking: Boolean(page.chunkSize),\n chunkSize: page.chunkSize * 1e6, // this option expects Bytes\n parallelChunkUploads: false, // for now, enabling this breaks descriptive upload progress\n timeout: 0,\n\n init () {\n this.on('addedfile', file => {\n // Set active tab to file uploads, if necessary\n if (page.activeTab !== 0)\n page.setActiveTab(0)\n\n // Add file entry\n tabDiv.querySelector('.uploads').classList.remove('is-hidden')\n\n file.previewElement.querySelector('.name').innerHTML = file.name\n file.previewElement.querySelector('.descriptive-progress').innerHTML = 'Waiting in queue\\u2026'\n })\n\n this.on('sending', (file, xhr) => {\n // Add timeout listener (hacky method due to lack of built-in timeout handler)\n if (!xhr.ontimeout)\n xhr.ontimeout = () => {\n const instances = page.dropzone.getUploadingFiles()\n .filter(instance => instance.xhr === xhr)\n page.dropzone._handleUploadError(instances, xhr, 'Connection timed out. Try to reduce upload chunk size.')\n }\n\n // Add start timestamp of upload attempt\n if (xhr._start === undefined)\n xhr._start = Date.now()\n\n // If not chunked uploads, add extra headers\n if (!file.upload.chunked) {\n if (page.album !== null) xhr.setRequestHeader('albumid', page.album)\n if (page.fileLength !== null) xhr.setRequestHeader('filelength', page.fileLength)\n if (page.uploadAge !== null) xhr.setRequestHeader('age', page.uploadAge)\n if (page.stripTags !== null) xhr.setRequestHeader('striptags', page.stripTags)\n }\n\n if (!file.upload.chunked)\n file.previewElement.querySelector('.descriptive-progress').innerHTML = 'Uploading\\u2026'\n else if (file.upload.chunks.length === 1)\n file.previewElement.querySelector('.descriptive-progress').innerHTML = `Uploading chunk 1/${file.upload.totalChunkCount}\\u2026`\n })\n\n // Update descriptive progress\n this.on('uploadprogress', (file, progress) => {\n // Total bytes will eventually be bigger than file size when chunked\n const total = Math.max(file.size, file.upload.total)\n const percentage = (file.upload.bytesSent / total * 100).toFixed(0)\n\n const upl = file.upload.chunked\n ? file.upload.chunks[file.upload.chunks.length - 1]\n : file.upload\n const xhr = upl.xhr || file.xhr\n\n let prefix = 'Uploading\\u2026'\n let skipProgress = false\n if (file.upload.chunked) {\n const done = upl.bytesSent === upl.total\n const last = file.upload.chunks.length === file.upload.totalChunkCount\n let chunkIndex = file.upload.chunks.length\n if (done && !last) {\n chunkIndex++\n skipProgress = true\n }\n prefix = `Uploading chunk ${chunkIndex}/${file.upload.totalChunkCount}\\u2026`\n }\n\n let prettyBytesPerSec\n if (!skipProgress) {\n const elapsed = (Date.now() - xhr._start) / 1000\n const bytesPerSec = elapsed ? (upl.bytesSent / elapsed) : 0\n prettyBytesPerSec = page.getPrettyBytes(bytesPerSec)\n }\n\n file.previewElement.querySelector('.descriptive-progress').innerHTML =\n `${prefix} ${percentage}%${prettyBytesPerSec ? ` at ~${prettyBytesPerSec}/s` : ''}`\n })\n\n this.on('success', (file, data) => {\n if (!data) return\n file.previewElement.querySelector('.descriptive-progress').classList.add('is-hidden')\n\n if (data.success === false) {\n file.previewElement.querySelector('.error').innerHTML = data.description\n file.previewElement.querySelector('.error').classList.remove('is-hidden')\n }\n\n if (Array.isArray(data.files) && data.files[0])\n page.updateTemplate(file, data.files[0])\n })\n\n this.on('error', (file, error) => {\n // Clean up file size errors\n if ((typeof error === 'string' && /^File is too big/.test(error)) ||\n (typeof error === 'object' && /File too large/.test(error.description)))\n error = `File too large (${page.getPrettyBytes(file.size)}).`\n\n page.updateTemplateIcon(file.previewElement, 'icon-block')\n\n file.previewElement.querySelector('.descriptive-progress').classList.add('is-hidden')\n\n file.previewElement.querySelector('.error').innerHTML = error.description || error\n file.previewElement.querySelector('.error').classList.remove('is-hidden')\n })\n },\n\n chunksUploaded (file, done) {\n file.previewElement.querySelector('.descriptive-progress').innerHTML =\n `Rebuilding ${file.upload.totalChunkCount} chunks\\u2026`\n\n return axios.post('api/upload/finishchunks', {\n // This API supports an array of multiple files\n files: [{\n uuid: file.upload.uuid,\n original: file.name,\n type: file.type,\n albumid: page.album,\n filelength: page.fileLength,\n age: page.uploadAge\n }]\n }, {\n headers: {\n token: page.token,\n // Unlike the options above (e.g. albumid, filelength, etc.),\n // strip tags can not yet be configured per file with this API\n striptags: page.stripTags\n }\n }).catch(error => {\n // Format error for display purpose\n return error.response.data ? error.response : {\n data: {\n success: false,\n description: error.toString()\n }\n }\n }).then(response => {\n file.previewElement.querySelector('.descriptive-progress').classList.add('is-hidden')\n\n if (response.data.success === false) {\n file.previewElement.querySelector('.error').innerHTML = response.data.description\n file.previewElement.querySelector('.error').classList.remove('is-hidden')\n }\n\n if (response.data.files && response.data.files[0])\n page.updateTemplate(file, response.data.files[0])\n\n return done()\n })\n }\n })\n}\n\npage.addUrlsToQueue = () => {\n const urls = document.querySelector('#urls').value\n .split(/\\r?\\n/)\n .filter(url => {\n return url.trim().length\n })\n\n if (!urls.length)\n return swal('An error occurred!', 'You have not entered any URLs.', 'error')\n\n const tabDiv = document.querySelector('#tab-urls')\n tabDiv.querySelector('.uploads').classList.remove('is-hidden')\n\n for (let i = 0; i < urls.length; i++) {\n const previewTemplate = document.createElement('template')\n previewTemplate.innerHTML = page.previewTemplate.trim()\n\n const previewElement = previewTemplate.content.firstChild\n previewElement.querySelector('.name').innerHTML = urls[i]\n previewElement.querySelector('.descriptive-progress').innerHTML = 'Waiting in queue\\u2026'\n\n const previewsContainer = tabDiv.querySelector('.uploads')\n previewsContainer.appendChild(previewElement)\n\n page.urlsQueue.push({\n url: urls[i],\n previewElement\n })\n }\n\n page.processUrlsQueue()\n document.querySelector('#urls').value = ''\n}\n\npage.processUrlsQueue = () => {\n if (!page.urlsQueue.length) return\n\n function finishedUrlUpload (file, data) {\n file.previewElement.querySelector('.descriptive-progress').classList.add('is-hidden')\n\n if (data.success === false) {\n const match = data.description.match(/ over limit: (\\d+)$/)\n if (match && match[1])\n data.description = `File exceeded limit of ${page.getPrettyBytes(match[1])}.`\n\n file.previewElement.querySelector('.error').innerHTML = data.description\n file.previewElement.querySelector('.error').classList.remove('is-hidden')\n }\n\n if (Array.isArray(data.files) && data.files[0])\n page.updateTemplate(file, data.files[0])\n\n page.activeUrlsQueue--\n return shiftQueue()\n }\n\n function initUrlUpload (file) {\n file.previewElement.querySelector('.descriptive-progress').innerHTML =\n 'Waiting for server to fetch URL\\u2026'\n\n return axios.post('api/upload', {\n urls: [file.url]\n }, {\n headers: {\n token: page.token,\n albumid: page.album,\n age: page.uploadAge,\n filelength: page.fileLength\n }\n }).catch(error => {\n // Format error for display purpose\n return error.response.data ? error.response : {\n data: {\n success: false,\n description: error.toString()\n }\n }\n }).then(response => {\n return finishedUrlUpload(file, response.data)\n })\n }\n\n function shiftQueue () {\n while (page.urlsQueue.length && (page.activeUrlsQueue < page.parallelUploads)) {\n page.activeUrlsQueue++\n initUrlUpload(page.urlsQueue.shift())\n }\n }\n\n return shiftQueue()\n}\n\npage.updateTemplateIcon = (templateElement, iconClass) => {\n const iconElement = templateElement.querySelector('.icon')\n if (!iconElement) return\n\n iconElement.classList.add(iconClass)\n iconElement.classList.remove('is-hidden')\n}\n\npage.updateTemplate = (file, response) => {\n if (!response.url) return\n\n const link = file.previewElement.querySelector('.link')\n const a = link.querySelector('a')\n const clipboard = file.previewElement.querySelector('.clipboard-mobile > .clipboard-js')\n a.href = a.innerHTML = clipboard.dataset.clipboardText = response.url\n\n link.classList.remove('is-hidden')\n clipboard.parentElement.classList.remove('is-hidden')\n\n const exec = /.[\\w]+(\\?|$)/.exec(response.url)\n const extname = exec && exec[0]\n ? exec[0].toLowerCase()\n : null\n\n if (page.imageExts.includes(extname))\n if (page.previewImages) {\n const img = file.previewElement.querySelector('img')\n img.setAttribute('alt', response.name || '')\n img.dataset.src = response.url\n img.classList.remove('is-hidden')\n img.onerror = event => {\n // Hide image elements that fail to load\n // Consequently include WEBP in browsers that do not have WEBP support (e.g. IE)\n event.currentTarget.classList.add('is-hidden')\n page.updateTemplateIcon(file.previewElement, 'icon-picture')\n }\n page.lazyLoad.update(file.previewElement.querySelectorAll('img'))\n } else {\n page.updateTemplateIcon(file.previewElement, 'icon-picture')\n }\n else if (page.videoExts.includes(extname))\n page.updateTemplateIcon(file.previewElement, 'icon-video')\n else\n page.updateTemplateIcon(file.previewElement, 'icon-doc-inv')\n\n if (response.expirydate) {\n const expiryDate = file.previewElement.querySelector('.expiry-date')\n expiryDate.innerHTML = `EXP: ${page.getPrettyDate(new Date(response.expirydate * 1000))}`\n expiryDate.classList.remove('is-hidden')\n }\n}\n\npage.createAlbum = () => {\n const div = document.createElement('div')\n div.innerHTML = `\n <div class=\"field\">\n <div class=\"controls\">\n <input id=\"swalName\" class=\"input\" type=\"text\" placeholder=\"Name\" maxlength=\"${page.albumTitleMaxLength}\">\n </div>\n <p class=\"help\">Max length is ${page.albumTitleMaxLength} characters.</p>\n </div>\n <div class=\"field\">\n <div class=\"control\">\n <textarea id=\"swalDescription\" class=\"textarea\" placeholder=\"Description\" rows=\"2\" maxlength=\"${page.albumDescMaxLength}\"></textarea>\n </div>\n <p class=\"help\">Max length is ${page.albumDescMaxLength} characters.</p>\n </div>\n <div class=\"field\">\n <div class=\"control\">\n <label class=\"checkbox\">\n <input id=\"swalDownload\" type=\"checkbox\" checked>\n Enable download\n </label>\n </div>\n </div>\n <div class=\"field\">\n <div class=\"control\">\n <label class=\"checkbox\">\n <input id=\"swalPublic\" type=\"checkbox\" checked>\n Enable public link\n </label>\n </div>\n </div>\n `\n\n swal({\n title: 'Create new album',\n icon: 'info',\n content: div,\n buttons: {\n cancel: true,\n confirm: {\n closeModal: false\n }\n }\n }).then(value => {\n if (!value) return\n\n const name = document.querySelector('#swalName').value.trim()\n axios.post('api/albums', {\n name,\n description: document.querySelector('#swalDescription').value.trim(),\n download: document.querySelector('#swalDownload').checked,\n public: document.querySelector('#swalPublic').checked\n }, {\n headers: {\n token: page.token\n }\n }).then(response => {\n if (response.data.success === false)\n return swal('An error occurred!', response.data.description, 'error')\n\n const option = document.createElement('option')\n page.albumSelect.appendChild(option)\n option.value = response.data.id\n option.innerHTML = name\n option.selected = true\n page.albumSelectOnChange()\n\n swal('Woohoo!', 'Album was created successfully.', 'success')\n }).catch(page.onError)\n })\n}\n\npage.prepareUploadConfig = () => {\n const fallback = {\n chunkSize: page.chunkSize,\n parallelUploads: page.parallelUploads\n }\n\n const temporaryUploadAges = Array.isArray(page.temporaryUploadAges) && page.temporaryUploadAges.length\n const fileIdentifierLength = page.fileIdentifierLength &&\n typeof page.fileIdentifierLength.min === 'number' &&\n typeof page.fileIdentifierLength.max === 'number'\n\n const config = {\n siBytes: {\n label: 'File size display',\n select: [\n { value: 'default', text: '1000 B = 1 kB = 1 Kilobyte' },\n { value: '0', text: '1024 B = 1 KiB = 1 Kibibyte' }\n ],\n help: 'This will be used in our homepage, dashboard, and album public pages.',\n valueHandler () {} // Do nothing\n },\n fileLength: {\n display: fileIdentifierLength,\n label: 'File identifier length',\n number: fileIdentifierLength ? {\n min: page.fileIdentifierLength.min,\n max: page.fileIdentifierLength.max,\n round: true\n } : undefined,\n help: true, // true means auto-generated, for number-based configs only\n disabled: fileIdentifierLength && page.fileIdentifierLength.force\n },\n uploadAge: {\n display: temporaryUploadAges,\n label: 'Upload age',\n select: [],\n help: 'Whether to automatically delete your uploads after a certain amount of time.'\n },\n stripTags: {\n display: page.stripTagsConfig,\n label: 'Strip tags',\n select: page.stripTagsConfig ? [\n { value: page.stripTagsConfig.default ? 'default' : '1', text: 'Yes' },\n { value: page.stripTagsConfig.default ? '0' : 'default', text: 'No' }\n ] : null,\n help: `Whether to strip tags (e.g. EXIF) from your uploads.<br>\n This only applies to regular image${page.stripTagsConfig && page.stripTagsConfig.video ? ' and video' : ''} uploads (i.e. not URL uploads).`,\n disabled: page.stripTagsConfig && page.stripTagsConfig.force\n },\n chunkSize: {\n display: !isNaN(page.chunkSize),\n label: 'Upload chunk size (MB)',\n number: {\n min: 1,\n max: 95,\n suffix: ' MB',\n round: true\n },\n help: true\n },\n parallelUploads: {\n label: 'Parallel uploads',\n number: {\n min: 1,\n max: 10,\n round: true\n },\n help: true\n },\n uploadsHistoryOrder: {\n label: 'Uploads history order',\n select: [\n { value: 'default', text: 'Older files on top' },\n { value: '0', text: 'Newer files on top' }\n ],\n help: `\"Newer files on top\" will use a CSS technique, which unfortunately come with <a href=\"https://developer.mozilla.org/en-US/docs/Web/CSS/flex-direction#Accessibility_Concerns\" target=\"_blank\" rel=\"noopener\">some undesirable side effects</a>.<br>\n This also affects text selection, such as when trying to select text from top to bottom will result in them being selected from bottom to top instead, and vice versa.`,\n valueHandler (value) {\n if (value === '0') {\n const uploadFields = document.querySelectorAll('.tab-content > .uploads')\n for (let i = 0; i < uploadFields.length; i++)\n uploadFields[i].classList.add('is-reversed')\n }\n }\n },\n previewImages: {\n label: 'Load images for preview',\n select: [\n { value: 'default', text: 'Yes' },\n { value: '0', text: 'No' }\n ],\n help: 'By default, uploaded images will be loaded as their previews.',\n valueHandler (value) {\n page.previewImages = value !== '0'\n }\n }\n }\n\n if (temporaryUploadAges) {\n const stored = parseFloat(localStorage[lsKeys.uploadAge])\n for (let i = 0; i < page.temporaryUploadAges.length; i++) {\n const age = page.temporaryUploadAges[i]\n config.uploadAge.select.push({\n value: i === 0 ? 'default' : String(age),\n text: page.getPrettyUploadAge(age)\n })\n if (age === stored)\n config.uploadAge.value = stored\n }\n }\n\n if (fileIdentifierLength) {\n fallback.fileLength = page.fileIdentifierLength.default || undefined\n const stored = parseInt(localStorage[lsKeys.fileLength])\n if (!page.fileIdentifierLength.force &&\n !isNaN(stored) &&\n stored >= page.fileIdentifierLength.min &&\n stored <= page.fileIdentifierLength.max)\n config.fileLength.value = stored\n }\n\n const tabContent = document.querySelector('#tab-config')\n const form = document.createElement('form')\n form.addEventListener('submit', event => event.preventDefault())\n\n const configKeys = Object.keys(config)\n for (let i = 0; i < configKeys.length; i++) {\n const key = configKeys[i]\n const conf = config[key]\n\n // Skip only if display attribute is explicitly set to false\n if (conf.display === false)\n continue\n\n const field = document.createElement('div')\n field.className = 'field'\n\n let value\n if (!conf.disabled) {\n if (conf.value !== undefined) {\n value = conf.value\n } else if (conf.number !== undefined) {\n const parsed = parseInt(localStorage[lsKeys[key]])\n if (!isNaN(parsed))\n value = parsed\n } else {\n const stored = localStorage[lsKeys[key]]\n if (Array.isArray(conf.select))\n value = conf.select.find(sel => sel.value === stored) ? stored : undefined\n else\n value = stored\n }\n\n // If valueHandler function exists, defer to the function,\n // otherwise pass value to global page object\n if (typeof conf.valueHandler === 'function')\n conf.valueHandler(value)\n else if (value !== undefined)\n page[key] = value\n }\n\n let control\n if (Array.isArray(conf.select)) {\n control = document.createElement('div')\n control.className = 'select is-fullwidth'\n\n const opts = []\n for (let j = 0; j < conf.select.length; j++) {\n const opt = conf.select[j]\n const selected = (value && (opt.value === String(value))) ||\n (value === undefined && opt.value === 'default')\n opts.push(`\n <option value=\"${opt.value}\"${selected ? ' selected' : ''}>\n ${opt.text}${opt.value === 'default' ? ' (default)' : ''}\n </option>\n `)\n }\n\n control.innerHTML = `\n <select id=\"${key}\">\n ${opts.join('\\n')}\n </select>\n `\n } else if (conf.number !== undefined) {\n control = document.createElement('input')\n control.id = control.name = key\n control.className = 'input is-fullwidth'\n control.type = 'number'\n\n if (conf.number.min !== undefined)\n control.min = conf.number.min\n if (conf.number.max !== undefined)\n control.max = conf.number.max\n if (typeof value === 'number')\n control.value = value\n else if (fallback[key] !== undefined)\n control.value = fallback[key]\n }\n\n let help\n if (conf.disabled) {\n if (Array.isArray(conf.select))\n control.querySelector('select').disabled = conf.disabled\n else\n control.disabled = conf.disabled\n help = 'This option is currently not configurable.'\n } else if (typeof conf.help === 'string') {\n help = conf.help\n } else if (conf.help === true && conf.number !== undefined) {\n const tmp = []\n\n if (fallback[key] !== undefined)\n tmp.push(`Default is ${fallback[key]}${conf.number.suffix || ''}.`)\n if (conf.number.min !== undefined)\n tmp.push(`Min is ${conf.number.min}${conf.number.suffix || ''}.`)\n if (conf.number.max !== undefined)\n tmp.push(`Max is ${conf.number.max}${conf.number.suffix || ''}.`)\n\n help = tmp.join(' ')\n }\n\n field.innerHTML = `\n <label class=\"label\">${conf.label}</label>\n <div class=\"control\"></div>\n ${help ? `<p class=\"help\">${help}</p>` : ''}\n `\n field.querySelector('div.control').appendChild(control)\n\n form.appendChild(field)\n }\n\n const submit = document.createElement('div')\n submit.className = 'field'\n submit.innerHTML = `\n <p class=\"control\">\n <button id=\"saveConfig\" type=\"submit\" class=\"button is-danger is-outlined is-fullwidth\">\n <span class=\"icon\">\n <i class=\"icon-floppy\"></i>\n </span>\n <span>Save & reload</span>\n </button>\n </p>\n <p class=\"help\">\n This configuration will only be used in this browser.<br>\n After reloading the page, some of them will also be applied to the ShareX config that you can download by clicking on the ShareX icon below.\n </p>\n `\n\n form.appendChild(submit)\n form.querySelector('#saveConfig').addEventListener('click', () => {\n if (!form.checkValidity())\n return\n\n const keys = Object.keys(config)\n .filter(key => config[key].display !== false && config[key].disabled !== true)\n for (let i = 0; i < keys.length; i++) {\n const key = keys[i]\n\n let value\n if (config[key].select !== undefined) {\n if (form.elements[key].value !== 'default')\n value = form.elements[key].value\n } else if (config[key].number !== undefined) {\n const parsed = parseInt(form.elements[key].value)\n if (!isNaN(parsed))\n value = Math.min(Math.max(parsed, config[key].number.min), config[key].number.max)\n }\n\n if (value !== undefined && value !== fallback[key])\n localStorage[lsKeys[key]] = value\n else\n localStorage.removeItem(lsKeys[key])\n }\n\n swal({\n title: 'Woohoo!',\n text: 'Configuration saved into this browser.',\n icon: 'success'\n }).then(() => {\n location.reload()\n })\n })\n\n tabContent.appendChild(form)\n}\n\npage.getPrettyUploadAge = hours => {\n if (hours === 0) {\n return 'Permanent'\n } else if (hours < 1) {\n const minutes = hours * 60\n return `${minutes} minute${minutes === 1 ? '' : 's'}`\n } else if (hours >= 24) {\n const days = hours / 24\n return `${days} day${days === 1 ? '' : 's'}`\n } else {\n return `${hours} hour${hours === 1 ? '' : 's'}`\n }\n}\n\n// Handle image paste event\nwindow.addEventListener('paste', event => {\n const items = (event.clipboardData || event.originalEvent.clipboardData).items\n const index = Object.keys(items)\n for (let i = 0; i < index.length; i++) {\n const item = items[index[i]]\n if (item.kind === 'file') {\n const blob = item.getAsFile()\n /* eslint-disable-next-line compat/compat */\n const file = new File([blob], `pasted-image.${blob.type.match(/(?:[^/]*\\/)([^;]*)/)[1]}`, {\n type: blob.type\n })\n page.dropzone.addFile(file)\n }\n }\n})\n\nwindow.onload = () => {\n page.checkIfPublic()\n\n page.clipboardJS = new ClipboardJS('.clipboard-js')\n\n page.clipboardJS.on('success', () => {\n return swal('', 'The link has been copied to clipboard.', 'success', {\n buttons: false,\n timer: 1500\n })\n })\n\n page.clipboardJS.on('error', page.onError)\n\n page.lazyLoad = new LazyLoad({\n elements_selector: '.field.uploads img'\n })\n\n document.querySelector('#createAlbum').addEventListener('click', () => {\n page.createAlbum()\n })\n}\n"]} |