FileItem.js 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. const getFileNameAndExtension = require('@uppy/utils/lib/getFileNameAndExtension')
  2. const truncateString = require('./truncateString')
  3. const copyToClipboard = require('./copyToClipboard')
  4. const prettyBytes = require('prettier-bytes')
  5. const FileItemProgress = require('./FileItemProgress')
  6. const getFileTypeIcon = require('./getFileTypeIcon')
  7. const FilePreview = require('./FilePreview')
  8. const { iconEdit, iconCopy, iconRetry } = require('./icons')
  9. const classNames = require('classnames')
  10. const { h } = require('preact')
  11. const FileItemProgressWrapper = (props) => {
  12. if (props.hideRetryButton && props.error) {
  13. return
  14. }
  15. if (props.isUploaded ||
  16. props.bundled ||
  17. (props.hidePauseResumeCancelButtons && !props.error)) {
  18. return <div class="uppy-DashboardItem-progressIndicator">
  19. <FileItemProgress
  20. progress={props.file.progress.percentage}
  21. fileID={props.file.id}
  22. hidePauseResumeCancelButtons={props.hidePauseResumeCancelButtons}
  23. bundled={props.bundled}
  24. />
  25. </div>
  26. }
  27. return <button
  28. class="uppy-DashboardItem-progressIndicator"
  29. type="button"
  30. aria-label={props.progressIndicatorTitle}
  31. title={props.progressIndicatorTitle}
  32. onclick={props.onPauseResumeCancelRetry}>
  33. {props.error
  34. ? props.hideRetryButton ? null : iconRetry()
  35. : <FileItemProgress
  36. progress={props.file.progress.percentage}
  37. fileID={props.file.id}
  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, 14) : fileName
  54. const 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 {
  66. props.cancelUpload(file.id)
  67. }
  68. }
  69. const dashboardItemClass = classNames(
  70. 'uppy-DashboardItem',
  71. { 'is-inprogress': uploadInProgress },
  72. { 'is-processing': isProcessing },
  73. { 'is-complete': isUploaded },
  74. { 'is-paused': isPaused },
  75. { 'is-error': error },
  76. { 'is-resumable': props.resumableUploads },
  77. { 'is-bundled': props.bundledUpload }
  78. )
  79. const progressIndicatorTitle = isUploaded
  80. ? props.i18n('uploadComplete')
  81. : props.resumableUploads
  82. ? file.isPaused
  83. ? props.i18n('resumeUpload')
  84. : props.i18n('pauseUpload')
  85. : error
  86. ? props.i18n('retryUpload')
  87. : props.i18n('cancelUpload')
  88. return <li class={dashboardItemClass} id={`uppy_${file.id}`} title={file.meta.name}>
  89. <div class="uppy-DashboardItem-preview">
  90. <div class="uppy-DashboardItem-previewInnerWrap" style={{ backgroundColor: getFileTypeIcon(file.type).color }}>
  91. {props.showLinkToFileUploadResult && file.uploadURL
  92. ? <a class="uppy-DashboardItem-previewLink" href={file.uploadURL} rel="noreferrer noopener" target="_blank" />
  93. : null
  94. }
  95. <FilePreview file={file} />
  96. </div>
  97. <div class="uppy-DashboardItem-progress">
  98. <FileItemProgressWrapper
  99. progressIndicatorTitle={progressIndicatorTitle}
  100. onPauseResumeCancelRetry={onPauseResumeCancelRetry}
  101. file={file}
  102. error={error}
  103. {...props} />
  104. </div>
  105. </div>
  106. <div class="uppy-DashboardItem-info">
  107. <div class="uppy-DashboardItem-name" title={fileName}>
  108. {props.showLinkToFileUploadResult && file.uploadURL
  109. ? <a href={file.uploadURL} rel="noreferrer noopener" target="_blank">
  110. {file.extension ? truncatedFileName + '.' + file.extension : truncatedFileName}
  111. </a>
  112. : file.extension ? truncatedFileName + '.' + file.extension : truncatedFileName
  113. }
  114. </div>
  115. <div class="uppy-DashboardItem-status">
  116. {file.data.size ? <div class="uppy-DashboardItem-statusSize">{prettyBytes(file.data.size)}</div> : null}
  117. {file.source && <div class="uppy-DashboardItem-sourceIcon">
  118. {acquirers.map(acquirer => {
  119. if (acquirer.id === file.source) {
  120. return <span title={props.i18n('fileSource', { name: acquirer.name })}>
  121. {acquirer.icon()}
  122. </span>
  123. }
  124. })}
  125. </div>
  126. }
  127. </div>
  128. {(!uploadInProgressOrComplete && props.metaFields && props.metaFields.length)
  129. ? <button class="uppy-DashboardItem-edit"
  130. type="button"
  131. aria-label={props.i18n('editFile')}
  132. title={props.i18n('editFile')}
  133. onclick={(e) => props.toggleFileCard(file.id)}>
  134. {iconEdit()}
  135. </button>
  136. : null
  137. }
  138. {props.showLinkToFileUploadResult && file.uploadURL
  139. ? <button class="uppy-DashboardItem-copyLink"
  140. type="button"
  141. aria-label={props.i18n('copyLink')}
  142. title={props.i18n('copyLink')}
  143. onclick={() => {
  144. copyToClipboard(file.uploadURL, props.i18n('copyLinkToClipboardFallback'))
  145. .then(() => {
  146. props.log('Link copied to clipboard.')
  147. props.info(props.i18n('copyLinkToClipboardSuccess'), 'info', 3000)
  148. })
  149. .catch(props.log)
  150. }}>{iconCopy()}</button>
  151. : ''
  152. }
  153. </div>
  154. <div class="uppy-DashboardItem-action">
  155. {!isUploaded &&
  156. <button class="uppy-DashboardItem-remove"
  157. type="button"
  158. aria-label={props.i18n('removeFile')}
  159. title={props.i18n('removeFile')}
  160. onclick={() => props.removeFile(file.id)}>
  161. <svg aria-hidden="true" class="UppyIcon" width="60" height="60" viewBox="0 0 60 60" xmlns="http://www.w3.org/2000/svg">
  162. <path stroke="#FFF" stroke-width="1" fill-rule="nonzero" vector-effect="non-scaling-stroke" d="M30 1C14 1 1 14 1 30s13 29 29 29 29-13 29-29S46 1 30 1z" />
  163. <path fill="#FFF" vector-effect="non-scaling-stroke" d="M42 39.667L39.667 42 30 32.333 20.333 42 18 39.667 27.667 30 18 20.333 20.333 18 30 27.667 39.667 18 42 20.333 32.333 30z" />
  164. </svg>
  165. </button>
  166. }
  167. </div>
  168. </li>
  169. }