2024-06-21 07:40:33 +00:00
import { svg } from '../svg.js' ;
import { htmlEscape } from 'escape-goat' ;
import { clippie } from 'clippie' ;
import { showTemporaryTooltip } from '../modules/tippy.js' ;
2024-06-26 17:01:20 +00:00
import { GET , POST } from '../modules/fetch.js' ;
2024-06-21 07:40:33 +00:00
import { showErrorToast } from '../modules/toast.js' ;
2024-06-26 17:01:20 +00:00
import { createElementFromHTML , createElementFromAttrs } from '../utils/dom.js' ;
2024-06-21 07:40:33 +00:00
const { csrfToken , i18n } = window . config ;
2024-06-26 17:01:20 +00:00
async function createDropzone ( el , opts ) {
2021-04-09 02:32:30 +00:00
const [ { Dropzone } ] = await Promise . all ( [
2020-03-07 21:06:15 +00:00
import ( /* webpackChunkName: "dropzone" */ 'dropzone' ) ,
import ( /* webpackChunkName: "dropzone" */ 'dropzone/dist/dropzone.css' ) ,
] ) ;
return new Dropzone ( el , opts ) ;
}
2024-06-21 07:40:33 +00:00
2024-06-26 17:01:20 +00:00
function addCopyLink ( file ) {
// Create a "Copy Link" element, to conveniently copy the image or file link as Markdown to the clipboard
// The "<a>" element has a hardcoded cursor: pointer because the default is overridden by .dropzone
const copyLinkEl = createElementFromHTML ( `
< div class = "tw-text-center" >
< a href = "#" class = "tw-cursor-pointer" > $ { svg ( 'octicon-copy' , 14 ) } Copy link < / a >
< / d i v > ` ) ;
copyLinkEl . addEventListener ( 'click' , async ( e ) => {
e . preventDefault ( ) ;
let fileMarkdown = ` [ ${ file . name } ](/attachments/ ${ file . uuid } ) ` ;
if ( file . type ? . startsWith ( 'image/' ) ) {
fileMarkdown = ` ! ${ fileMarkdown } ` ;
} else if ( file . type ? . startsWith ( 'video/' ) ) {
fileMarkdown = ` <video src="/attachments/ ${ htmlEscape ( file . uuid ) } " title=" ${ htmlEscape ( file . name ) } " controls></video> ` ;
}
const success = await clippie ( fileMarkdown ) ;
showTemporaryTooltip ( e . target , success ? i18n . copy _success : i18n . copy _error ) ;
} ) ;
file . previewTemplate . append ( copyLinkEl ) ;
2024-06-21 07:40:33 +00:00
}
2024-06-26 17:01:20 +00:00
/ * *
* @ param { HTMLElement } dropzoneEl
* /
export async function initDropzone ( dropzoneEl ) {
const listAttachmentsUrl = dropzoneEl . closest ( '[data-attachment-url]' ) ? . getAttribute ( 'data-attachment-url' ) ;
const removeAttachmentUrl = dropzoneEl . getAttribute ( 'data-remove-url' ) ;
const attachmentBaseLinkUrl = dropzoneEl . getAttribute ( 'data-link-url' ) ;
let disableRemovedfileEvent = false ; // when resetting the dropzone (removeAllFiles), disable the "removedfile" event
let fileUuidDict = { } ; // to record: if a comment has been saved, then the uploaded files won't be deleted from server when clicking the Remove in the dropzone
const opts = {
url : dropzoneEl . getAttribute ( 'data-upload-url' ) ,
2024-06-21 07:40:33 +00:00
headers : { 'X-Csrf-Token' : csrfToken } ,
2024-06-26 17:01:20 +00:00
acceptedFiles : [ '*/*' , '' ] . includes ( dropzoneEl . getAttribute ( 'data-accepts' ) ) ? null : dropzoneEl . getAttribute ( 'data-accepts' ) ,
2024-06-21 07:40:33 +00:00
addRemoveLinks : true ,
2024-06-26 17:01:20 +00:00
dictDefaultMessage : dropzoneEl . getAttribute ( 'data-default-message' ) ,
dictInvalidFileType : dropzoneEl . getAttribute ( 'data-invalid-input-type' ) ,
dictFileTooBig : dropzoneEl . getAttribute ( 'data-file-too-big' ) ,
dictRemoveFile : dropzoneEl . getAttribute ( 'data-remove-file' ) ,
2024-06-21 07:40:33 +00:00
timeout : 0 ,
thumbnailMethod : 'contain' ,
thumbnailWidth : 480 ,
thumbnailHeight : 480 ,
2024-06-26 17:01:20 +00:00
} ;
if ( dropzoneEl . hasAttribute ( 'data-max-file' ) ) opts . maxFiles = Number ( dropzoneEl . getAttribute ( 'data-max-file' ) ) ;
if ( dropzoneEl . hasAttribute ( 'data-max-size' ) ) opts . maxFilesize = Number ( dropzoneEl . getAttribute ( 'data-max-size' ) ) ;
// there is a bug in dropzone: if a non-image file is uploaded, then it tries to request the file from server by something like:
// "http://localhost:3000/owner/repo/issues/[object%20Event]"
// the reason is that the preview "callback(dataURL)" is assign to "img.onerror" then "thumbnail" uses the error object as the dataURL and generates '<img src="[object Event]">'
const dzInst = await createDropzone ( dropzoneEl , opts ) ;
dzInst . on ( 'success' , ( file , data ) => {
file . uuid = data . uuid ;
fileUuidDict [ file . uuid ] = { submitted : false } ;
const input = createElementFromAttrs ( 'input' , { name : 'files' , type : 'hidden' , id : ` dropzone-file- ${ data . uuid } ` , value : data . uuid } ) ;
dropzoneEl . querySelector ( '.files' ) . append ( input ) ;
addCopyLink ( file ) ;
} ) ;
dzInst . on ( 'removedfile' , async ( file ) => {
if ( disableRemovedfileEvent ) return ;
document . querySelector ( ` #dropzone-file- ${ file . uuid } ` ) ? . remove ( ) ;
// when the uploaded file number reaches the limit, there is no uuid in the dict, and it doesn't need to be removed from server
if ( removeAttachmentUrl && fileUuidDict [ file . uuid ] && ! fileUuidDict [ file . uuid ] . submitted ) {
await POST ( removeAttachmentUrl , { data : new URLSearchParams ( { file : file . uuid } ) } ) ;
}
} ) ;
dzInst . on ( 'submit' , ( ) => {
for ( const fileUuid of Object . keys ( fileUuidDict ) ) {
fileUuidDict [ fileUuid ] . submitted = true ;
}
2024-06-21 07:40:33 +00:00
} ) ;
2024-06-26 17:01:20 +00:00
dzInst . on ( 'reload' , async ( ) => {
try {
const resp = await GET ( listAttachmentsUrl ) ;
const respData = await resp . json ( ) ;
// do not trigger the "removedfile" event, otherwise the attachments would be deleted from server
disableRemovedfileEvent = true ;
dzInst . removeAllFiles ( true ) ;
disableRemovedfileEvent = false ;
dropzoneEl . querySelector ( '.files' ) . innerHTML = '' ;
for ( const el of dropzoneEl . querySelectorAll ( '.dz-preview' ) ) el . remove ( ) ;
fileUuidDict = { } ;
for ( const attachment of respData ) {
const imgSrc = ` ${ attachmentBaseLinkUrl } / ${ attachment . uuid } ` ;
dzInst . emit ( 'addedfile' , attachment ) ;
dzInst . emit ( 'thumbnail' , attachment , imgSrc ) ;
dzInst . emit ( 'complete' , attachment ) ;
addCopyLink ( attachment ) ;
fileUuidDict [ attachment . uuid ] = { submitted : true } ;
const input = createElementFromAttrs ( 'input' , { name : 'files' , type : 'hidden' , id : ` dropzone-file- ${ attachment . uuid } ` , value : attachment . uuid } ) ;
dropzoneEl . querySelector ( '.files' ) . append ( input ) ;
}
if ( ! dropzoneEl . querySelector ( '.dz-preview' ) ) {
dropzoneEl . classList . remove ( 'dz-started' ) ;
}
} catch ( error ) {
// TODO: if listing the existing attachments failed, it should stop from operating the content or attachments,
// otherwise the attachments might be lost.
showErrorToast ( ` Failed to load attachments: ${ error } ` ) ;
console . error ( error ) ;
}
} ) ;
dzInst . on ( 'error' , ( file , message ) => {
showErrorToast ( ` Dropzone upload error: ${ message } ` ) ;
dzInst . removeFile ( file ) ;
} ) ;
if ( listAttachmentsUrl ) dzInst . emit ( 'reload' ) ;
return dzInst ;
2024-06-21 07:40:33 +00:00
}