FileItem.js 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. const getFileNameAndExtension = require('@uppy/utils/lib/getFileNameAndExtension')
  2. const truncateString = require('../utils/truncateString')
  3. const copyToClipboard = require('../utils/copyToClipboard')
  4. const prettyBytes = require('prettier-bytes')
  5. const FileItemProgress = require('./FileItemProgress')
  6. const getFileTypeIcon = require('../utils/getFileTypeIcon')
  7. const FilePreview = require('./FilePreview')
  8. const { iconRetry } = require('./icons')
  9. const classNames = require('classnames')
  10. const { h } = require('preact')
  11. function FileItemProgressWrapper (props) {
  12. if (props.hideRetryButton && props.error) {
  13. return
  14. }
  15. if (props.isUploaded ||
  16. (props.hidePauseResumeCancelButtons && !props.error)) {
  17. return <div class="uppy-DashboardItem-progressIndicator">
  18. <FileItemProgress
  19. progress={props.file.progress.percentage}
  20. fileID={props.file.id}
  21. hidePauseResumeCancelButtons={props.hidePauseResumeCancelButtons}
  22. individualCancellation={props.individualCancellation}
  23. />
  24. </div>
  25. }
  26. return <button
  27. class="uppy-DashboardItem-progressIndicator"
  28. type="button"
  29. aria-label={props.progressIndicatorTitle}
  30. title={props.progressIndicatorTitle}
  31. onclick={props.onPauseResumeCancelRetry}>
  32. {props.error
  33. ? props.hideRetryButton ? null : iconRetry()
  34. : <FileItemProgress
  35. progress={props.file.progress.percentage}
  36. fileID={props.file.id}
  37. individualCancellation={props.individualCancellation}
  38. hidePauseResumeCancelButtons={props.hidePauseResumeCancelButtons}
  39. />
  40. }
  41. </button>
  42. }
  43. module.exports = function FileItem (props) {
  44. const file = props.file
  45. const acquirers = props.acquirers
  46. const isProcessing = file.progress.preprocess || file.progress.postprocess
  47. const isUploaded = file.progress.uploadComplete && !isProcessing && !file.error
  48. const uploadInProgressOrComplete = file.progress.uploadStarted || isProcessing
  49. const uploadInProgress = (file.progress.uploadStarted && !file.progress.uploadComplete) || isProcessing
  50. const isPaused = file.isPaused || false
  51. const error = file.error || false
  52. const fileName = getFileNameAndExtension(file.meta.name).name
  53. const truncatedFileName = props.isWide ? truncateString(fileName, 30) : fileName
  54. function onPauseResumeCancelRetry (ev) {
  55. if (isUploaded) return
  56. if (error && !props.hideRetryButton) {
  57. props.retryUpload(file.id)
  58. return
  59. }
  60. if (props.hidePauseResumeCancelButtons) {
  61. return
  62. }
  63. if (props.resumableUploads) {
  64. props.pauseUpload(file.id)
  65. } else if (props.individualCancellation) {
  66. props.cancelUpload(file.id)
  67. }
  68. }
  69. function progressIndicatorTitle (props) {
  70. if (isUploaded) {
  71. return props.i18n('uploadComplete')
  72. }
  73. if (error) {
  74. return props.i18n('retryUpload')
  75. }
  76. if (props.resumableUploads) {
  77. if (file.isPaused) {
  78. return props.i18n('resumeUpload')
  79. }
  80. return props.i18n('pauseUpload')
  81. } else if (props.individualCancellation) {
  82. return props.i18n('cancelUpload')
  83. }
  84. return ''
  85. }
  86. const dashboardItemClass = classNames(
  87. 'uppy-DashboardItem',
  88. { 'is-inprogress': uploadInProgress },
  89. { 'is-processing': isProcessing },
  90. { 'is-complete': isUploaded },
  91. { 'is-paused': isPaused },
  92. { 'is-error': error },
  93. { 'is-resumable': props.resumableUploads },
  94. { 'is-noIndividualCancellation': !props.individualCancellation }
  95. )
  96. const showRemoveButton = props.individualCancellation
  97. ? !isUploaded
  98. : !uploadInProgress && !isUploaded
  99. return <li class={dashboardItemClass} id={`uppy_${file.id}`}>
  100. <div class="uppy-DashboardItem-preview">
  101. <div class="uppy-DashboardItem-previewInnerWrap" style={{ backgroundColor: getFileTypeIcon(file.type).color }}>
  102. {props.showLinkToFileUploadResult && file.uploadURL
  103. ? <a class="uppy-DashboardItem-previewLink" href={file.uploadURL} rel="noreferrer noopener" target="_blank" aria-label={file.meta.name} />
  104. : null
  105. }
  106. <FilePreview file={file} />
  107. </div>
  108. <div class="uppy-DashboardItem-progress">
  109. <FileItemProgressWrapper
  110. progressIndicatorTitle={progressIndicatorTitle(props)}
  111. onPauseResumeCancelRetry={onPauseResumeCancelRetry}
  112. file={file}
  113. error={error}
  114. isUploaded={isUploaded}
  115. {...props} />
  116. </div>
  117. </div>
  118. <div class="uppy-DashboardItem-info">
  119. <div class="uppy-DashboardItem-name">
  120. {file.extension ? truncatedFileName + '.' + file.extension : truncatedFileName}
  121. </div>
  122. <div class="uppy-DashboardItem-status">
  123. {
  124. file.data.size && [
  125. <div class="uppy-DashboardItem-statusSize">
  126. {prettyBytes(file.data.size)}
  127. </div>,
  128. <span class="uppy-DashboardItem-statusbar-dot">·</span>
  129. ]
  130. }
  131. {
  132. (file.source && file.source !== props.id) && [
  133. <div class="uppy-DashboardItem-sourceIcon">
  134. {acquirers.map(acquirer => {
  135. if (acquirer.id === file.source) {
  136. return <span title={props.i18n('fileSource', { name: acquirer.name })}>
  137. {acquirer.icon()}
  138. </span>
  139. }
  140. })}
  141. </div>,
  142. <span class="uppy-DashboardItem-statusbar-dot">·</span>
  143. ]
  144. }
  145. {
  146. (!uploadInProgressOrComplete && props.metaFields && props.metaFields.length) && [
  147. <button class="uppy-u-reset uppy-DashboardItem-edit"
  148. type="button"
  149. aria-label={props.i18n('editFile') + ' ' + fileName}
  150. title={props.i18n('editFile')}
  151. onclick={(e) => props.toggleFileCard(file.id)}>
  152. {props.i18n('edit')}
  153. </button>,
  154. <span class="uppy-DashboardItem-statusbar-dot">·</span>
  155. ]
  156. }
  157. {props.showLinkToFileUploadResult && file.uploadURL
  158. ? <button class="uppy-u-reset uppy-DashboardItem-copyLink"
  159. type="button"
  160. aria-label={props.i18n('copyLink')}
  161. title={props.i18n('copyLink')}
  162. onclick={() => {
  163. copyToClipboard(file.uploadURL, props.i18n('copyLinkToClipboardFallback'))
  164. .then(() => {
  165. props.log('Link copied to clipboard.')
  166. props.info(props.i18n('copyLinkToClipboardSuccess'), 'info', 3000)
  167. })
  168. .catch(props.log)
  169. }}>{props.i18n('link')}</button>
  170. : ''
  171. }
  172. </div>
  173. </div>
  174. <div class="uppy-DashboardItem-action">
  175. {showRemoveButton &&
  176. <button class="uppy-DashboardItem-remove"
  177. type="button"
  178. aria-label={props.i18n('removeFile')}
  179. title={props.i18n('removeFile')}
  180. onclick={() => props.removeFile(file.id)}>
  181. <svg aria-hidden="true" class="UppyIcon" width="18" height="18" viewBox="0 0 18 18">
  182. <path d="M9 0C4.034 0 0 4.034 0 9s4.034 9 9 9 9-4.034 9-9-4.034-9-9-9z" />
  183. <path fill="#FFF" d="M13 12.222l-.778.778L9 9.778 5.778 13 5 12.222 8.222 9 5 5.778 5.778 5 9 8.222 12.222 5l.778.778L9.778 9z" />
  184. </svg>
  185. </button>
  186. }
  187. </div>
  188. </li>
  189. }