cardDetails.jade 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577
  1. template(name="cardDetails")
  2. section.card-details.js-card-details.js-perfect-scrollbar: .card-details-canvas
  3. .card-details-header(class='{{#if colorClass}}card-details-{{colorClass}}{{/if}}')
  4. +inlinedForm(classNames="js-card-details-title")
  5. +editCardTitleForm
  6. else
  7. unless isMiniScreen
  8. a.fa.fa-times-thin.close-card-details.js-close-card-details
  9. if currentUser.isBoardMember
  10. a.fa.fa-navicon.card-details-menu.js-open-card-details-menu
  11. input.inline-input(type="text" id="cardURL_copy" value="{{ absoluteUrl }}")
  12. a.fa.fa-link.card-copy-button.js-copy-link(
  13. class="fa-link"
  14. title="{{_ 'copy-card-link-to-clipboard'}}"
  15. value="{{ absoluteUrl }}"
  16. )
  17. if isMiniScreen
  18. a.fa.fa-times-thin.close-card-details-mobile-web.js-close-card-details
  19. if currentUser.isBoardMember
  20. a.fa.fa-navicon.card-details-menu-mobile-web.js-open-card-details-menu
  21. a.fa.fa-link.card-copy-mobile-button
  22. h2.card-details-title.js-card-title(
  23. class="{{#if canModifyCard}}js-open-inlined-form is-editable{{/if}}")
  24. +viewer
  25. = getTitle
  26. if isWatching
  27. i.card-details-watch.fa.fa-eye
  28. .card-details-path
  29. each parentList
  30. |   >  
  31. a.js-parent-card(href=linkForCard) {{title}}
  32. // else
  33. {{_ 'top-level-card'}}
  34. if isLinkedCard
  35. h3.linked-card-location
  36. +viewer
  37. | {{getBoardTitle}} > {{getTitle}}
  38. if getArchived
  39. if isLinkedBoard
  40. p.warning {{_ 'board-archived'}}
  41. else
  42. p.warning {{_ 'card-archived'}}
  43. .card-details-items
  44. if currentBoard.allowsReceivedDate
  45. .card-details-item.card-details-item-received
  46. h3
  47. i.fa.fa-sign-out
  48. card-details-item-title {{_ 'card-received'}}
  49. if getReceived
  50. +cardReceivedDate
  51. else
  52. if canModifyCard
  53. unless currentUser.isWorker
  54. a.card-label.add-label.js-received-date
  55. i.fa.fa-plus
  56. if currentBoard.allowsStartDate
  57. .card-details-item.card-details-item-start
  58. h3
  59. i.fa.fa-hourglass-start
  60. card-details-item-title {{_ 'card-start'}}
  61. if getStart
  62. +cardStartDate
  63. else
  64. if canModifyCard
  65. unless currentUser.isWorker
  66. a.card-label.add-label.js-start-date
  67. i.fa.fa-plus
  68. if currentBoard.allowsDueDate
  69. .card-details-item.card-details-item-due
  70. h3
  71. i.fa.fa-sign-in
  72. card-details-item-title {{_ 'card-due'}}
  73. if getDue
  74. +cardDueDate
  75. else
  76. if canModifyCard
  77. unless currentUser.isWorker
  78. a.card-label.add-label.js-due-date
  79. i.fa.fa-plus
  80. if currentBoard.allowsEndDate
  81. .card-details-item.card-details-item-end
  82. h3
  83. i.fa.fa-hourglass-end
  84. card-details-item-title {{_ 'card-end'}}
  85. if getEnd
  86. +cardEndDate
  87. else
  88. if canModifyCard
  89. unless currentUser.isWorker
  90. a.card-label.add-label.js-end-date
  91. i.fa.fa-plus
  92. //.card-details-items
  93. if currentBoard.allowsMembers
  94. .card-details-item.card-details-item-members
  95. h3
  96. i.fa.fa-users
  97. card-details-item-title {{_ 'members'}}
  98. each getMembers
  99. +userAvatar(userId=this cardId=../_id)
  100. | {{! XXX Hack to hide syntaxic coloration /// }}
  101. if canModifyCard
  102. unless currentUser.isWorker
  103. a.member.add-member.card-details-item-add-button.js-add-members(title="{{_ 'card-members-title'}}")
  104. i.fa.fa-plus
  105. //if assigneeSelected
  106. if currentBoard.allowsAssignee
  107. .card-details-item.card-details-item-assignees
  108. h3
  109. i.fa.fa-user
  110. card-details-item-title {{_ 'assignee'}}
  111. each getAssignees
  112. +userAvatarAssignee(userId=this cardId=../_id)
  113. | {{! XXX Hack to hide syntaxic coloration /// }}
  114. if canModifyCard
  115. a.assignee.add-assignee.card-details-item-add-button.js-add-assignees(title="{{_ 'assignee'}}")
  116. i.fa.fa-plus
  117. if currentUser.isWorker
  118. unless assigneeSelected
  119. a.assignee.add-assignee.card-details-item-add-button.js-add-assignees(title="{{_ 'assignee'}}")
  120. i.fa.fa-plus
  121. if currentBoard.allowsLabels
  122. .card-details-item.card-details-item-labels
  123. h3
  124. i.fa.fa-tags
  125. card-details-item-title {{_ 'labels'}}
  126. a(class="{{#if canModifyCard}}js-add-labels{{else}}is-disabled{{/if}}" title="{{_ 'card-labels-title'}}")
  127. each labels
  128. span.card-label(class="card-label-{{color}}" title=name)
  129. +viewer
  130. = name
  131. if canModifyCard
  132. unless currentUser.isWorker
  133. a.card-label.add-label.js-add-labels(title="{{_ 'card-labels-title'}}")
  134. i.fa.fa-plus
  135. //.card-details-items
  136. each customFieldsWD
  137. .card-details-item.card-details-item-customfield
  138. h3.card-details-item-title
  139. +viewer
  140. = definition.name
  141. +cardCustomField
  142. //.card-details-items
  143. if getSpentTime
  144. .card-details-item.card-details-item-spent
  145. if getIsOvertime
  146. h3.card-details-item-title {{_ 'overtime-hours'}}
  147. else
  148. h3.card-details-item-title {{_ 'spent-time-hours'}}
  149. +cardSpentTime
  150. //.card-details-items
  151. if currentBoard.allowsRequestedBy
  152. .card-details-item.card-details-item-name
  153. h3
  154. i.fa.fa-shopping-cart
  155. card-details-item-title {{_ 'requested-by'}}
  156. if canModifyCard
  157. unless currentUser.isWorker
  158. +inlinedForm(classNames="js-card-details-requester")
  159. +editCardRequesterForm
  160. else
  161. a.js-open-inlined-form
  162. if getRequestedBy
  163. +viewer
  164. = getRequestedBy
  165. else
  166. | {{_ 'add'}}
  167. else if getRequestedBy
  168. +viewer
  169. = getRequestedBy
  170. if currentBoard.allowsAssignedBy
  171. .card-details-item.card-details-item-name
  172. h3
  173. i.fa.fa-user-plus
  174. card-details-item-title {{_ 'assigned-by'}}
  175. if canModifyCard
  176. unless currentUser.isWorker
  177. +inlinedForm(classNames="js-card-details-assigner")
  178. +editCardAssignerForm
  179. else
  180. a.js-open-inlined-form
  181. if getAssignedBy
  182. +viewer
  183. = getAssignedBy
  184. else
  185. | {{_ 'add'}}
  186. else if getRequestedBy
  187. +viewer
  188. = getAssignedBy
  189. if getVoteQuestion
  190. hr
  191. .vote-title
  192. h3
  193. i.fa.fa-thumbs-up
  194. card-details-item-title {{_ 'vote-question'}}
  195. .vote-result
  196. .card-label.card-label-green
  197. +viewer
  198. = voteCountPositive
  199. .card-label.card-label-red
  200. +viewer
  201. = voteCountNegative
  202. +viewer
  203. = getVoteQuestion
  204. button.card-details-green.js-vote.js-vote-positive(class="{{#if voteState}}voted{{/if}}") {{_ 'vote-for-it'}}
  205. button.card-details-red.js-vote.js-vote-negative(class="{{#if $eq voteState false}}voted{{/if}}") {{_ 'vote-against'}}
  206. //- XXX We should use "editable" to avoid repetiting ourselves
  207. if canModifyCard
  208. unless currentUser.isWorker
  209. if currentBoard.allowsDescriptionTitle
  210. hr
  211. h3
  212. i.fa.fa-align-left
  213. card-details-item-title {{_ 'description'}}
  214. if currentBoard.allowsDescriptionText
  215. +inlinedCardDescription(classNames="card-description js-card-description")
  216. +editor(autofocus=true)
  217. | {{getUnsavedValue 'cardDescription' _id getDescription}}
  218. .edit-controls.clearfix
  219. button.primary(type="submit") {{_ 'save'}}
  220. a.fa.fa-times-thin.js-close-inlined-form
  221. else
  222. if currentBoard.allowsDescriptionText
  223. a.js-open-inlined-form
  224. if getDescription
  225. +viewer
  226. = getDescription
  227. else
  228. | {{_ 'edit'}}
  229. if (hasUnsavedValue 'cardDescription' _id)
  230. p.quiet
  231. | {{_ 'unsaved-description'}}
  232. a.js-open-inlined-form {{_ 'view-it'}}
  233. = ' - '
  234. a.js-close-inlined-form {{_ 'discard'}}
  235. else if getDescription
  236. if currentBoard.allowsDescriptionTitle
  237. hr
  238. h3.card-details-item-title {{_ 'description'}}
  239. if currentBoard.allowsDescriptionText
  240. +viewer
  241. = getDescription
  242. .card-checklist-attachmentGalerys
  243. .card-checklist-attachmentGalery.card-checklists
  244. if currentBoard.allowsChecklists
  245. hr
  246. +checklists(cardId = _id)
  247. if currentBoard.allowsSubtasks
  248. hr
  249. +subtasks(cardId = _id)
  250. if currentBoard.allowsAttachments
  251. hr
  252. h3
  253. i.fa.fa-paperclip
  254. | {{_ 'attachments'}}
  255. .card-checklist-attachmentGalery.card-attachmentGalery
  256. +attachmentsGalery
  257. hr
  258. unless currentUser.isNoComments
  259. .activity-title
  260. h3
  261. i.fa.fa-history
  262. | {{ _ 'activity'}}
  263. if currentUser.isBoardMember
  264. .material-toggle-switch
  265. span.toggle-switch-title {{_ 'hide-system-messages'}}
  266. if hiddenSystemMessages
  267. input.toggle-switch(type="checkbox" id="toggleButton" checked="checked")
  268. else
  269. input.toggle-switch(type="checkbox" id="toggleButton")
  270. label.toggle-label(for="toggleButton")
  271. if currentBoard.allowsComments
  272. if currentUser.isBoardMember
  273. unless currentUser.isNoComments
  274. +commentForm
  275. unless currentUser.isNoComments
  276. if isLoaded.get
  277. if isLinkedCard
  278. +activities(card=this mode="linkedcard")
  279. else if isLinkedBoard
  280. +activities(card=this mode="linkedboard")
  281. else
  282. +activities(card=this mode="card")
  283. template(name="editCardTitleForm")
  284. textarea.js-edit-card-title(rows='1' autofocus dir="auto")
  285. = getTitle
  286. .edit-controls.clearfix
  287. button.primary.confirm.js-submit-edit-card-title-form(type="submit") {{_ 'save'}}
  288. a.fa.fa-times-thin.js-close-inlined-form
  289. template(name="editCardRequesterForm")
  290. input.js-edit-card-requester(type='text' autofocus value=getRequestedBy dir="auto")
  291. .edit-controls.clearfix
  292. button.primary.confirm.js-submit-edit-card-requester-form(type="submit") {{_ 'save'}}
  293. a.fa.fa-times-thin.js-close-inlined-form
  294. template(name="editCardAssignerForm")
  295. input.js-edit-card-assigner(type='text' autofocus value=getAssignedBy dir="auto")
  296. .edit-controls.clearfix
  297. button.primary.confirm.js-submit-edit-card-assigner-form(type="submit") {{_ 'save'}}
  298. a.fa.fa-times-thin.js-close-inlined-form
  299. template(name="cardDetailsActionsPopup")
  300. ul.pop-over-list
  301. li
  302. a.js-toggle-watch-card
  303. if isWatching
  304. i.fa.fa-eye
  305. | {{_ 'unwatch'}}
  306. else
  307. i.fa.fa-eye-slash
  308. | {{_ 'watch'}}
  309. if canModifyCard
  310. unless currentUser.isWorker
  311. hr
  312. ul.pop-over-list
  313. //li: a.js-members {{_ 'card-edit-members'}}
  314. //li: a.js-labels {{_ 'card-edit-labels'}}
  315. //li: a.js-attachments {{_ 'card-edit-attachments'}}
  316. if getVoteQuestion
  317. li
  318. a.js-cancel-voting
  319. i.fa.fa-thumbs-up
  320. | {{_ 'card-cancel-voting'}}
  321. else
  322. li
  323. a.js-start-voting
  324. i.fa.fa-thumbs-up
  325. | {{_ 'card-start-voting'}}
  326. li
  327. a.js-custom-fields
  328. i.fa.fa-list-alt
  329. | {{_ 'card-edit-custom-fields'}}
  330. //li: a.js-received-date {{_ 'editCardReceivedDatePopup-title'}}
  331. //li: a.js-start-date {{_ 'editCardStartDatePopup-title'}}
  332. //li: a.js-due-date {{_ 'editCardDueDatePopup-title'}}
  333. //li: a.js-end-date {{_ 'editCardEndDatePopup-title'}}
  334. li
  335. a.js-spent-time
  336. i.fa.fa-clock-o
  337. | {{_ 'editCardSpentTimePopup-title'}}
  338. li
  339. a.js-set-card-color
  340. i.fa.fa-paint-brush
  341. | {{_ 'setCardColorPopup-title'}}
  342. hr
  343. ul.pop-over-list
  344. li
  345. a.js-move-card-to-top
  346. i.fa.fa-arrow-up
  347. | {{_ 'moveCardToTop-title'}}
  348. li
  349. a.js-move-card-to-bottom
  350. i.fa.fa-arrow-down
  351. | {{_ 'moveCardToBottom-title'}}
  352. unless currentUser.isWorker
  353. hr
  354. ul.pop-over-list
  355. li
  356. a.js-move-card
  357. i.fa.fa-arrow-right
  358. | {{_ 'moveCardPopup-title'}}
  359. li
  360. a.js-copy-card
  361. i.fa.fa-copy
  362. | {{_ 'copyCardPopup-title'}}
  363. hr
  364. ul.pop-over-list
  365. li
  366. a.js-copy-checklist-cards
  367. i.fa.fa-list
  368. i.fa.fa-copy
  369. | {{_ 'copyChecklistToManyCardsPopup-title'}}
  370. unless archived
  371. hr
  372. ul.pop-over-list
  373. li
  374. a.js-archive
  375. i.fa.fa-arrow-right
  376. i.fa.fa-archive
  377. | {{_ 'archive-card'}}
  378. hr
  379. ul.pop-over-list
  380. li
  381. a.js-more
  382. i.fa.fa-link
  383. | {{_ 'cardMorePopup-title'}}
  384. template(name="moveCardPopup")
  385. +boardsAndLists
  386. template(name="copyCardPopup")
  387. label(for='copy-card-title') {{_ 'title'}}:
  388. textarea#copy-card-title.minicard-composer-textarea.js-card-title(autofocus)
  389. = getTitle
  390. +boardsAndLists
  391. template(name="copyChecklistToManyCardsPopup")
  392. label(for='copy-checklist-cards-title') {{_ 'copyChecklistToManyCardsPopup-instructions'}}:
  393. textarea#copy-card-title.minicard-composer-textarea.js-card-title(autofocus)
  394. | {{_ 'copyChecklistToManyCardsPopup-format'}}
  395. +boardsAndLists
  396. template(name="boardsAndLists")
  397. label {{_ 'boards'}}:
  398. select.js-select-boards(autofocus)
  399. each boards
  400. if $eq _id currentBoard._id
  401. option(value="{{_id}}" selected) {{_ 'current'}}
  402. else
  403. option(value="{{_id}}") {{title}}
  404. label {{_ 'swimlanes'}}:
  405. select.js-select-swimlanes
  406. each swimlanes
  407. option(value="{{_id}}") {{title}}
  408. label {{_ 'lists'}}:
  409. select.js-select-lists
  410. each aBoardLists
  411. option(value="{{_id}}") {{title}}
  412. .edit-controls.clearfix
  413. button.primary.confirm.js-done {{_ 'done'}}
  414. template(name="cardMembersPopup")
  415. ul.pop-over-list.js-card-member-list
  416. each board.activeMembers
  417. li.item(class="{{#if isCardMember}}active{{/if}}")
  418. a.name.js-select-member(href="#")
  419. +userAvatar(userId=user._id)
  420. span.full-name
  421. = user.profile.fullname
  422. | (<span class="username">{{ user.username }}</span>)
  423. if isCardMember
  424. i.fa.fa-check
  425. template(name="cardAssigneesPopup")
  426. unless currentUser.isWorker
  427. ul.pop-over-list.js-card-assignee-list
  428. each board.activeMembers
  429. li.item(class="{{#if isCardAssignee}}active{{/if}}")
  430. a.name.js-select-assignee(href="#")
  431. +userAvatar(userId=user._id)
  432. span.full-name
  433. = user.profile.fullname
  434. | (<span class="username">{{ user.username }}</span>)
  435. if isCardAssignee
  436. i.fa.fa-check
  437. if currentUser.isWorker
  438. ul.pop-over-list.js-card-assignee-list
  439. li.item(class="{{#if currentUser.isCardAssignee}}active{{/if}}")
  440. a.name.js-select-assignee(href="#")
  441. +userAvatar(userId=currentUser._id)
  442. span.full-name
  443. = currentUser.profile.fullname
  444. | (<span class="username">{{ currentUser.username }}</span>)
  445. if currentUser.isCardAssignee
  446. i.fa.fa-check
  447. template(name="userAvatarAssignee")
  448. a.assignee.js-assignee(title="{{userData.profile.fullname}} ({{userData.username}})")
  449. if userData.profile.avatarUrl
  450. img.avatar.avatar-image(src="{{userData.profile.avatarUrl}}")
  451. else
  452. +userAvatarAssigneeInitials(userId=userData._id)
  453. if showStatus
  454. span.assignee-presence-status(class=presenceStatusClassName)
  455. span.member-type(class=memberType)
  456. unless isSandstorm
  457. if showEdit
  458. if $eq currentUser._id userData._id
  459. a.edit-avatar.js-change-avatar
  460. i.fa.fa-pencil
  461. template(name="cardAssigneePopup")
  462. .board-assignee-menu
  463. .mini-profile-info
  464. +userAvatar(userId=user._id showEdit=true)
  465. .info
  466. h3= user.profile.fullname
  467. p.quiet @{{ user.username }}
  468. ul.pop-over-list
  469. if currentUser.isNotCommentOnly
  470. unless currentUser.isWorker
  471. li: a.js-remove-assignee {{_ 'remove-member-from-card'}}
  472. unless currentUser.isWorker
  473. if $eq currentUser._id user._id
  474. with currentUser
  475. li: a.js-edit-profile {{_ 'edit-profile'}}
  476. template(name="userAvatarAssigneeInitials")
  477. svg.avatar.avatar-assignee-initials(viewBox="0 0 {{viewPortWidth}} 15")
  478. text(x="50%" y="13" text-anchor="middle")= initials
  479. template(name="cardMorePopup")
  480. p.quiet
  481. span.clearfix
  482. span {{_ 'link-card'}}
  483. = ' '
  484. i.fa.colorful(class="{{#if board.isPublic}}fa-globe{{else}}fa-lock{{/if}}")
  485. input.inline-input(type="text" id="cardURL" readonly value="{{ absoluteUrl }}" autofocus="autofocus")
  486. button.js-copy-card-link-to-clipboard(class="btn") {{_ 'copy-card-link-to-clipboard'}}
  487. span.clearfix
  488. br
  489. h2 {{_ 'change-card-parent'}}
  490. label {{_ 'source-board'}}:
  491. select.js-field-parent-board
  492. if isTopLevel
  493. option(value="none" selected) {{_ 'custom-field-dropdown-none'}}
  494. else
  495. option(value="none") {{_ 'custom-field-dropdown-none'}}
  496. each boards
  497. if isParentBoard
  498. option(value="{{_id}}" selected) {{title}}
  499. else
  500. option(value="{{_id}}") {{title}}
  501. label {{_ 'parent-card'}}:
  502. select.js-field-parent-card
  503. if isTopLevel
  504. option(value="none" selected) {{_ 'custom-field-dropdown-none'}}
  505. else
  506. option(value="none") {{_ 'custom-field-dropdown-none'}}
  507. each cards
  508. if isParentCard
  509. option(value="{{_id}}" selected) {{title}}
  510. else
  511. option(value="{{_id}}") {{title}}
  512. br
  513. | {{_ 'added'}}
  514. span.date(title=card.createdAt) {{ moment createdAt 'LLL' }}
  515. a.js-delete(title="{{_ 'card-delete-notice'}}") {{_ 'delete'}}
  516. template(name="setCardColorPopup")
  517. form.edit-label
  518. .palette-colors: each colors
  519. unless $eq color 'white'
  520. span.card-label.palette-color.js-palette-color(class="card-details-{{color}}")
  521. if(isSelected color)
  522. i.fa.fa-check
  523. button.primary.confirm.js-submit {{_ 'save'}}
  524. button.js-remove-color.negate.wide.right {{_ 'unset-color'}}
  525. template(name="cardDeletePopup")
  526. p {{_ "card-delete-pop"}}
  527. unless archived
  528. p {{_ "card-delete-suggest-archive"}}
  529. button.js-confirm.negate.full(type="submit") {{_ 'delete'}}
  530. template(name="cardStartVotingPopup")
  531. form.edit-vote-question
  532. .fields
  533. label(for="vote") {{_ 'vote-question'}}
  534. input.js-vote-field#vote(type="text" name="vote" value="{{card.getVoteQuestion}}" autofocus)
  535. button.primary.confirm.js-submit {{_ 'save'}}
  536. //- button.js-remove-color.negate.wide.right {{_ 'delete'}}