minicard.js 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  1. import { ReactiveCache } from '/imports/reactiveCache';
  2. import { TAPi18n } from '/imports/i18n';
  3. import { CustomFieldStringTemplate } from '/client/lib/customFields';
  4. import { handleFileUpload } from './attachments';
  5. import uploadProgressManager from '../../lib/uploadProgressManager';
  6. // Template.cards.events({
  7. // 'click .member': Popup.open('cardMember')
  8. // });
  9. BlazeComponent.extendComponent({
  10. template() {
  11. return 'minicard';
  12. },
  13. formattedCurrencyCustomFieldValue(definition) {
  14. const customField = this.data()
  15. .customFieldsWD()
  16. .find(f => f._id === definition._id);
  17. const customFieldTrueValue =
  18. customField && customField.trueValue ? customField.trueValue : '';
  19. const locale = TAPi18n.getLanguage();
  20. return new Intl.NumberFormat(locale, {
  21. style: 'currency',
  22. currency: definition.settings.currencyCode,
  23. }).format(customFieldTrueValue);
  24. },
  25. formattedStringtemplateCustomFieldValue(definition) {
  26. const customField = this.data()
  27. .customFieldsWD()
  28. .find(f => f._id === definition._id);
  29. const customFieldTrueValue =
  30. customField && customField.trueValue ? customField.trueValue : [];
  31. const ret = new CustomFieldStringTemplate(definition).getFormattedValue(customFieldTrueValue);
  32. return ret;
  33. },
  34. showCreatorOnMinicard() {
  35. // cache "board" to reduce the mini-mongodb access
  36. const board = this.data().board();
  37. let ret = false;
  38. if (board) {
  39. ret = board.allowsCreatorOnMinicard ?? false;
  40. }
  41. return ret;
  42. },
  43. isWatching() {
  44. const card = this.currentData();
  45. return card.findWatcher(Meteor.userId());
  46. },
  47. showMembers() {
  48. // cache "board" to reduce the mini-mongodb access
  49. const board = this.data().board();
  50. let ret = false;
  51. if (board) {
  52. ret =
  53. board.allowsMembers === null ||
  54. board.allowsMembers === undefined ||
  55. board.allowsMembers
  56. ;
  57. }
  58. return ret;
  59. },
  60. showAssignee() {
  61. // cache "board" to reduce the mini-mongodb access
  62. const board = this.data().board();
  63. let ret = false;
  64. if (board) {
  65. ret =
  66. board.allowsAssignee === null ||
  67. board.allowsAssignee === undefined ||
  68. board.allowsAssignee
  69. ;
  70. }
  71. return ret;
  72. },
  73. /** opens the card label popup only if clicked onto a label
  74. * <li> this is necessary to have the data context of the minicard.
  75. * if .js-card-label is used at click event, then only the data context of the label itself is available at this.currentData()
  76. */
  77. cardLabelsPopup(event) {
  78. if (this.find('.js-card-label:hover')) {
  79. Popup.open("cardLabels")(event, {dataContextIfCurrentDataIsUndefined: this.currentData()});
  80. }
  81. },
  82. events() {
  83. return [
  84. {
  85. 'click .js-linked-link'() {
  86. if (this.data().isLinkedCard()) Utils.goCardId(this.data().linkedId);
  87. else if (this.data().isLinkedBoard())
  88. Utils.goBoardId(this.data().linkedId);
  89. },
  90. 'click .js-toggle-minicard-label-text'() {
  91. if (window.localStorage.getItem('hiddenMinicardLabelText')) {
  92. window.localStorage.removeItem('hiddenMinicardLabelText'); //true
  93. } else {
  94. window.localStorage.setItem('hiddenMinicardLabelText', 'true'); //true
  95. }
  96. },
  97. 'click span.badge-icon.fa.fa-sort, click span.badge-text.check-list-sort' : Popup.open("editCardSortOrder"),
  98. 'click .minicard-labels' : this.cardLabelsPopup,
  99. 'click .js-open-minicard-details-menu': Popup.open('minicardDetailsActions'),
  100. // Drag and drop file upload handlers
  101. 'dragover .minicard'(event) {
  102. // Only prevent default for file drags to avoid interfering with sortable
  103. const dataTransfer = event.originalEvent.dataTransfer;
  104. if (dataTransfer && dataTransfer.types && dataTransfer.types.includes('Files')) {
  105. event.preventDefault();
  106. event.stopPropagation();
  107. }
  108. },
  109. 'dragenter .minicard'(event) {
  110. const dataTransfer = event.originalEvent.dataTransfer;
  111. if (dataTransfer && dataTransfer.types && dataTransfer.types.includes('Files')) {
  112. event.preventDefault();
  113. event.stopPropagation();
  114. const card = this.data();
  115. const board = card.board();
  116. // Only allow drag-and-drop if user can modify card and board allows attachments
  117. if (Utils.canModifyCard() && board && board.allowsAttachments) {
  118. $(event.currentTarget).addClass('is-dragging-over');
  119. }
  120. }
  121. },
  122. 'dragleave .minicard'(event) {
  123. const dataTransfer = event.originalEvent.dataTransfer;
  124. if (dataTransfer && dataTransfer.types && dataTransfer.types.includes('Files')) {
  125. event.preventDefault();
  126. event.stopPropagation();
  127. $(event.currentTarget).removeClass('is-dragging-over');
  128. }
  129. },
  130. 'drop .minicard'(event) {
  131. const dataTransfer = event.originalEvent.dataTransfer;
  132. if (dataTransfer && dataTransfer.types && dataTransfer.types.includes('Files')) {
  133. event.preventDefault();
  134. event.stopPropagation();
  135. $(event.currentTarget).removeClass('is-dragging-over');
  136. const card = this.data();
  137. const board = card.board();
  138. // Check permissions
  139. if (!Utils.canModifyCard() || !board || !board.allowsAttachments) {
  140. return;
  141. }
  142. // Check if this is a file drop (not a card reorder)
  143. if (!dataTransfer.files || dataTransfer.files.length === 0) {
  144. return;
  145. }
  146. const files = dataTransfer.files;
  147. if (files && files.length > 0) {
  148. handleFileUpload(card, files);
  149. }
  150. }
  151. },
  152. }
  153. ];
  154. },
  155. }).register('minicard');
  156. Template.minicard.helpers({
  157. hiddenMinicardLabelText() {
  158. const currentUser = ReactiveCache.getCurrentUser();
  159. if (currentUser) {
  160. return (currentUser.profile || {}).hiddenMinicardLabelText;
  161. } else if (window.localStorage.getItem('hiddenMinicardLabelText')) {
  162. return true;
  163. } else {
  164. return false;
  165. }
  166. },
  167. // XXX resolve this nasty hack for https://github.com/veliovgroup/Meteor-Files/issues/763
  168. sess() {
  169. return Meteor.connection && Meteor.connection._lastSessionId
  170. ? Meteor.connection._lastSessionId
  171. : null;
  172. },
  173. isWatching() {
  174. return this.findWatcher(Meteor.userId());
  175. },
  176. // Upload progress helpers
  177. hasActiveUploads() {
  178. return uploadProgressManager.hasActiveUploads(this._id);
  179. },
  180. uploads() {
  181. return uploadProgressManager.getUploadsForCard(this._id);
  182. },
  183. uploadCount() {
  184. return uploadProgressManager.getUploadCountForCard(this._id);
  185. },
  186. listName() {
  187. const list = this.list();
  188. return list ? list.title : '';
  189. },
  190. shouldShowListOnMinicard() {
  191. // Show list name if either:
  192. // 1. Board-wide setting is enabled, OR
  193. // 2. This specific card has the setting enabled
  194. const currentBoard = this.currentBoard;
  195. if (!currentBoard) return false;
  196. return currentBoard.allowsShowListsOnMinicard || this.showListOnMinicard;
  197. }
  198. });
  199. BlazeComponent.extendComponent({
  200. events() {
  201. return [
  202. {
  203. 'keydown input.js-edit-card-sort-popup'(evt) {
  204. // enter = save
  205. if (evt.keyCode === 13) {
  206. this.find('button[type=submit]').click();
  207. }
  208. },
  209. 'click button.js-submit-edit-card-sort-popup'(event) {
  210. // save button pressed
  211. event.preventDefault();
  212. const sort = this.$('.js-edit-card-sort-popup')[0]
  213. .value
  214. .trim();
  215. if (!Number.isNaN(sort)) {
  216. let card = this.data();
  217. card.move(card.boardId, card.swimlaneId, card.listId, sort);
  218. Popup.back();
  219. }
  220. },
  221. }
  222. ]
  223. }
  224. }).register('editCardSortOrderPopup');
  225. Template.minicardDetailsActionsPopup.events({
  226. 'click .js-due-date': Popup.open('editCardDueDate'),
  227. 'click .js-move-card': Popup.open('moveCard'),
  228. 'click .js-copy-card': Popup.open('copyCard'),
  229. 'click .js-set-card-color': Popup.open('setCardColor'),
  230. 'click .js-add-labels': Popup.open('cardLabels'),
  231. 'click .js-link': Popup.open('linkCard'),
  232. 'click .js-move-card-to-top'(event) {
  233. event.preventDefault();
  234. const minOrder = this.getMinSort();
  235. this.move(this.boardId, this.swimlaneId, this.listId, minOrder - 1);
  236. Popup.back();
  237. },
  238. 'click .js-move-card-to-bottom'(event) {
  239. event.preventDefault();
  240. const maxOrder = this.getMaxSort();
  241. this.move(this.boardId, this.swimlaneId, this.listId, maxOrder + 1);
  242. Popup.back();
  243. },
  244. 'click .js-archive': Popup.afterConfirm('cardArchive', function () {
  245. Popup.close();
  246. this.archive();
  247. Utils.goBoardId(this.boardId);
  248. }),
  249. 'click .js-toggle-watch-card'() {
  250. const currentCard = this;
  251. const level = currentCard.findWatcher(Meteor.userId()) ? null : 'watching';
  252. Meteor.call('watch', 'card', currentCard._id, level, (err, ret) => {
  253. if (!err && ret) Popup.back();
  254. });
  255. },
  256. });