FileItem.js 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. const html = require('yo-yo')
  2. const { getETA,
  3. getSpeed,
  4. prettyETA,
  5. getFileNameAndExtension,
  6. truncateString,
  7. copyToClipboard } = require('../../core/Utils')
  8. const prettyBytes = require('prettier-bytes')
  9. const FileItemProgress = require('./FileItemProgress')
  10. const getFileTypeIcon = require('./getFileTypeIcon')
  11. const { iconEdit, iconCopy, iconRetry } = require('./icons')
  12. module.exports = function fileItem (props) {
  13. const file = props.file
  14. const acquirers = props.acquirers
  15. const isProcessing = file.progress.preprocess || file.progress.postprocess
  16. const isUploaded = file.progress.uploadComplete && !isProcessing && !file.error
  17. const uploadInProgressOrComplete = file.progress.uploadStarted || isProcessing
  18. const uploadInProgress = (file.progress.uploadStarted && !file.progress.uploadComplete) || isProcessing
  19. const isPaused = file.isPaused || false
  20. const error = file.error || false
  21. const fileName = getFileNameAndExtension(file.meta.name).name
  22. const truncatedFileName = props.isWide ? truncateString(fileName, 15) : fileName
  23. const onPauseResumeCancelRetry = (ev) => {
  24. if (isUploaded) return
  25. if (error) {
  26. props.retryUpload(file.id)
  27. return
  28. }
  29. if (props.resumableUploads) {
  30. props.pauseUpload(file.id)
  31. } else {
  32. props.cancelUpload(file.id)
  33. }
  34. }
  35. return html`<li class="UppyDashboardItem
  36. ${uploadInProgress ? 'is-inprogress' : ''}
  37. ${isProcessing ? 'is-processing' : ''}
  38. ${isUploaded ? 'is-complete' : ''}
  39. ${isPaused ? 'is-paused' : ''}
  40. ${error ? 'is-error' : ''}
  41. ${props.resumableUploads ? 'is-resumable' : ''}"
  42. id="uppy_${file.id}"
  43. title="${file.meta.name}">
  44. <div class="UppyDashboardItem-preview">
  45. <div class="UppyDashboardItem-previewInnerWrap" style="background-color: ${getFileTypeIcon(file.type).color}">
  46. ${file.preview
  47. ? html`<img alt="${file.name}" src="${file.preview}">`
  48. : html`<div class="UppyDashboardItem-previewIconWrap">
  49. <span class="UppyDashboardItem-previewIcon" style="color: ${getFileTypeIcon(file.type).color}">${getFileTypeIcon(file.type).icon}</span>
  50. <svg class="UppyDashboardItem-previewIconBg" width="72" height="93" viewBox="0 0 72 93"><g><path d="M24.08 5h38.922A2.997 2.997 0 0 1 66 8.003v74.994A2.997 2.997 0 0 1 63.004 86H8.996A2.998 2.998 0 0 1 6 83.01V22.234L24.08 5z" fill="#FFF"/><path d="M24 5L6 22.248h15.007A2.995 2.995 0 0 0 24 19.244V5z" fill="#E4E4E4"/></g></svg>
  51. </div>`
  52. }
  53. </div>
  54. <div class="UppyDashboardItem-progress">
  55. <button class="UppyDashboardItem-progressBtn"
  56. type="button"
  57. title="${isUploaded
  58. ? 'upload complete'
  59. : props.resumableUploads
  60. ? file.isPaused
  61. ? 'resume upload'
  62. : 'pause upload'
  63. : 'cancel upload'
  64. }"
  65. onclick=${onPauseResumeCancelRetry}>
  66. ${error
  67. ? iconRetry()
  68. : FileItemProgress({
  69. progress: file.progress.percentage,
  70. fileID: file.id
  71. })
  72. }
  73. </button>
  74. ${props.showProgressDetails
  75. ? html`<div class="UppyDashboardItem-progressInfo"
  76. title="${props.i18n('fileProgress')}"
  77. aria-label="${props.i18n('fileProgress')}">
  78. ${!file.isPaused && !isUploaded
  79. ? html`<span>${prettyETA(getETA(file.progress))} ・ ↑ ${prettyBytes(getSpeed(file.progress))}/s</span>`
  80. : null
  81. }
  82. </div>`
  83. : null
  84. }
  85. </div>
  86. </div>
  87. <div class="UppyDashboardItem-info">
  88. <h4 class="UppyDashboardItem-name" title="${fileName}">
  89. ${file.uploadURL
  90. ? html`<a href="${file.uploadURL}" target="_blank">
  91. ${file.extension ? truncatedFileName + '.' + file.extension : truncatedFileName}
  92. </a>`
  93. : file.extension ? truncatedFileName + '.' + file.extension : truncatedFileName
  94. }
  95. </h4>
  96. <div class="UppyDashboardItem-status">
  97. ${file.data.size && html`<div class="UppyDashboardItem-statusSize">${prettyBytes(file.data.size)}</div>`}
  98. ${file.source && html`<div class="UppyDashboardItem-sourceIcon">
  99. ${acquirers.map(acquirer => {
  100. if (acquirer.id === file.source) return html`<span title="${props.i18n('fileSource')}: ${acquirer.name}">${acquirer.icon()}</span>`
  101. })}
  102. </div>`
  103. }
  104. </div>
  105. ${!uploadInProgressOrComplete
  106. ? html`<button class="UppyDashboardItem-edit"
  107. type="button"
  108. aria-label="Edit file"
  109. title="Edit file"
  110. onclick=${(e) => props.showFileCard(file.id)}>
  111. ${iconEdit()}</button>`
  112. : null
  113. }
  114. ${file.uploadURL
  115. ? html`<button class="UppyDashboardItem-copyLink"
  116. type="button"
  117. aria-label="Copy link"
  118. title="Copy link"
  119. onclick=${() => {
  120. copyToClipboard(file.uploadURL, props.i18n('copyLinkToClipboardFallback'))
  121. .then(() => {
  122. props.log('Link copied to clipboard.')
  123. props.info(props.i18n('copyLinkToClipboardSuccess'), 'info', 3000)
  124. })
  125. .catch(props.log)
  126. }}>${iconCopy()}</button>`
  127. : null
  128. }
  129. </div>
  130. <div class="UppyDashboardItem-action">
  131. ${!isUploaded
  132. ? html`<button class="UppyDashboardItem-remove"
  133. type="button"
  134. aria-label="Remove file"
  135. title="Remove file"
  136. onclick=${() => props.removeFile(file.id)}>
  137. <svg class="UppyIcon" width="22" height="21" viewBox="0 0 18 17">
  138. <ellipse cx="8.62" cy="8.383" rx="8.62" ry="8.383"/>
  139. <path stroke="#FFF" fill="#FFF" d="M11 6.147L10.85 6 8.5 8.284 6.15 6 6 6.147 8.35 8.43 6 10.717l.15.146L8.5 8.578l2.35 2.284.15-.146L8.65 8.43z"/>
  140. </svg>
  141. </button>`
  142. : null
  143. }
  144. </div>
  145. </li>`
  146. }