checklists.js 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420
  1. Checklists = new Mongo.Collection('checklists');
  2. /**
  3. * A Checklist
  4. */
  5. Checklists.attachSchema(
  6. new SimpleSchema({
  7. cardId: {
  8. /**
  9. * The ID of the card the checklist is in
  10. */
  11. type: String,
  12. },
  13. title: {
  14. /**
  15. * the title of the checklist
  16. */
  17. type: String,
  18. defaultValue: 'Checklist',
  19. },
  20. finishedAt: {
  21. /**
  22. * When was the checklist finished
  23. */
  24. type: Date,
  25. optional: true,
  26. },
  27. createdAt: {
  28. /**
  29. * Creation date of the checklist
  30. */
  31. type: Date,
  32. denyUpdate: false,
  33. // eslint-disable-next-line consistent-return
  34. autoValue() {
  35. if (this.isInsert) {
  36. return new Date();
  37. } else if (this.isUpsert) {
  38. return { $setOnInsert: new Date() };
  39. } else {
  40. this.unset();
  41. }
  42. },
  43. },
  44. modifiedAt: {
  45. type: Date,
  46. denyUpdate: false,
  47. // eslint-disable-next-line consistent-return
  48. autoValue() {
  49. if (this.isInsert || this.isUpsert || this.isUpdate) {
  50. return new Date();
  51. } else {
  52. this.unset();
  53. }
  54. },
  55. },
  56. sort: {
  57. /**
  58. * sorting value of the checklist
  59. */
  60. type: Number,
  61. decimal: true,
  62. },
  63. }),
  64. );
  65. Checklists.helpers({
  66. copy(newCardId) {
  67. const oldChecklistId = this._id;
  68. this._id = null;
  69. this.cardId = newCardId;
  70. const newChecklistId = Checklists.insert(this);
  71. ChecklistItems.find({ checklistId: oldChecklistId }).forEach(function(
  72. item,
  73. ) {
  74. item._id = null;
  75. item.checklistId = newChecklistId;
  76. item.cardId = newCardId;
  77. ChecklistItems.insert(item);
  78. });
  79. },
  80. itemCount() {
  81. return ChecklistItems.find({ checklistId: this._id }).count();
  82. },
  83. items() {
  84. return ChecklistItems.find(
  85. {
  86. checklistId: this._id,
  87. },
  88. { sort: ['sort'] },
  89. );
  90. },
  91. firstItem() {
  92. const allItems = this.items().fetch();
  93. const ret = _.first(allItems);
  94. return ret;
  95. },
  96. lastItem() {
  97. const allItems = this.items().fetch();
  98. const ret = allItems[allItems.length - 1];
  99. return ret;
  100. },
  101. finishedCount() {
  102. return ChecklistItems.find({
  103. checklistId: this._id,
  104. isFinished: true,
  105. }).count();
  106. },
  107. /** returns the finished percent of the checklist */
  108. finishedPercent() {
  109. const checklistItems = ChecklistItems.find({ checklistId: this._id });
  110. const count = checklistItems.count();
  111. const checklistItemsFinished = checklistItems.fetch().filter(checklistItem => checklistItem.isFinished);
  112. let ret = 0;
  113. if (count > 0) {
  114. ret = Math.round(checklistItemsFinished.length / count * 100);
  115. }
  116. return ret;
  117. },
  118. isFinished() {
  119. return 0 !== this.itemCount() && this.itemCount() === this.finishedCount();
  120. },
  121. checkAllItems() {
  122. const checkItems = ChecklistItems.find({ checklistId: this._id });
  123. checkItems.forEach(function(item) {
  124. item.check();
  125. });
  126. },
  127. uncheckAllItems() {
  128. const checkItems = ChecklistItems.find({ checklistId: this._id });
  129. checkItems.forEach(function(item) {
  130. item.uncheck();
  131. });
  132. },
  133. itemIndex(itemId) {
  134. const items = self.findOne({ _id: this._id }).items;
  135. return _.pluck(items, '_id').indexOf(itemId);
  136. },
  137. });
  138. Checklists.allow({
  139. insert(userId, doc) {
  140. return allowIsBoardMemberByCard(userId, Cards.findOne(doc.cardId));
  141. },
  142. update(userId, doc) {
  143. return allowIsBoardMemberByCard(userId, Cards.findOne(doc.cardId));
  144. },
  145. remove(userId, doc) {
  146. return allowIsBoardMemberByCard(userId, Cards.findOne(doc.cardId));
  147. },
  148. fetch: ['userId', 'cardId'],
  149. });
  150. Checklists.before.insert((userId, doc) => {
  151. doc.createdAt = new Date();
  152. if (!doc.userId) {
  153. doc.userId = userId;
  154. }
  155. });
  156. Checklists.mutations({
  157. setTitle(title) {
  158. return { $set: { title } };
  159. },
  160. /** move the checklist to another card
  161. * @param newCardId move the checklist to this cardId
  162. */
  163. move(newCardId) {
  164. // update every activity
  165. Activities.find(
  166. {checklistId: this._id}
  167. ).forEach(activity => {
  168. Activities.update(activity._id, {
  169. $set: {
  170. cardId: newCardId,
  171. },
  172. });
  173. });
  174. // update every checklist-item
  175. ChecklistItems.find(
  176. {checklistId: this._id}
  177. ).forEach(checklistItem => {
  178. ChecklistItems.update(checklistItem._id, {
  179. $set: {
  180. cardId: newCardId,
  181. },
  182. });
  183. });
  184. // update the checklist itself
  185. return {
  186. $set: {
  187. cardId: newCardId,
  188. },
  189. };
  190. },
  191. });
  192. if (Meteor.isServer) {
  193. Meteor.startup(() => {
  194. Checklists._collection.createIndex({ modifiedAt: -1 });
  195. Checklists._collection.createIndex({ cardId: 1, createdAt: 1 });
  196. });
  197. Checklists.after.insert((userId, doc) => {
  198. const card = Cards.findOne(doc.cardId);
  199. Activities.insert({
  200. userId,
  201. activityType: 'addChecklist',
  202. cardId: doc.cardId,
  203. boardId: card.boardId,
  204. checklistId: doc._id,
  205. checklistName: doc.title,
  206. listId: card.listId,
  207. swimlaneId: card.swimlaneId,
  208. });
  209. });
  210. Checklists.before.remove((userId, doc) => {
  211. const activities = Activities.find({ checklistId: doc._id });
  212. const card = Cards.findOne(doc.cardId);
  213. if (activities) {
  214. activities.forEach(activity => {
  215. Activities.remove(activity._id);
  216. });
  217. }
  218. Activities.insert({
  219. userId,
  220. activityType: 'removeChecklist',
  221. cardId: doc.cardId,
  222. boardId: Cards.findOne(doc.cardId).boardId,
  223. checklistId: doc._id,
  224. checklistName: doc.title,
  225. listId: card.listId,
  226. swimlaneId: card.swimlaneId,
  227. });
  228. });
  229. }
  230. if (Meteor.isServer) {
  231. /**
  232. * @operation get_all_checklists
  233. * @summary Get the list of checklists attached to a card
  234. *
  235. * @param {string} boardId the board ID
  236. * @param {string} cardId the card ID
  237. * @return_type [{_id: string,
  238. * title: string}]
  239. */
  240. JsonRoutes.add(
  241. 'GET',
  242. '/api/boards/:boardId/cards/:cardId/checklists',
  243. function(req, res) {
  244. const paramBoardId = req.params.boardId;
  245. const paramCardId = req.params.cardId;
  246. Authentication.checkBoardAccess(req.userId, paramBoardId);
  247. const checklists = Checklists.find({ cardId: paramCardId }).map(function(
  248. doc,
  249. ) {
  250. return {
  251. _id: doc._id,
  252. title: doc.title,
  253. };
  254. });
  255. if (checklists) {
  256. JsonRoutes.sendResult(res, {
  257. code: 200,
  258. data: checklists,
  259. });
  260. } else {
  261. JsonRoutes.sendResult(res, {
  262. code: 500,
  263. });
  264. }
  265. },
  266. );
  267. /**
  268. * @operation get_checklist
  269. * @summary Get a checklist
  270. *
  271. * @param {string} boardId the board ID
  272. * @param {string} cardId the card ID
  273. * @param {string} checklistId the ID of the checklist
  274. * @return_type {cardId: string,
  275. * title: string,
  276. * finishedAt: string,
  277. * createdAt: string,
  278. * sort: number,
  279. * items: [{_id: string,
  280. * title: string,
  281. * isFinished: boolean}]}
  282. */
  283. JsonRoutes.add(
  284. 'GET',
  285. '/api/boards/:boardId/cards/:cardId/checklists/:checklistId',
  286. function(req, res) {
  287. const paramBoardId = req.params.boardId;
  288. const paramChecklistId = req.params.checklistId;
  289. const paramCardId = req.params.cardId;
  290. Authentication.checkBoardAccess(req.userId, paramBoardId);
  291. const checklist = Checklists.findOne({
  292. _id: paramChecklistId,
  293. cardId: paramCardId,
  294. });
  295. if (checklist) {
  296. checklist.items = ChecklistItems.find({
  297. checklistId: checklist._id,
  298. }).map(function(doc) {
  299. return {
  300. _id: doc._id,
  301. title: doc.title,
  302. isFinished: doc.isFinished,
  303. };
  304. });
  305. JsonRoutes.sendResult(res, {
  306. code: 200,
  307. data: checklist,
  308. });
  309. } else {
  310. JsonRoutes.sendResult(res, {
  311. code: 500,
  312. });
  313. }
  314. },
  315. );
  316. /**
  317. * @operation new_checklist
  318. * @summary create a new checklist
  319. *
  320. * @param {string} boardId the board ID
  321. * @param {string} cardId the card ID
  322. * @param {string} title the title of the new checklist
  323. * @param {string} [items] the list of items on the new checklist
  324. * @return_type {_id: string}
  325. */
  326. JsonRoutes.add(
  327. 'POST',
  328. '/api/boards/:boardId/cards/:cardId/checklists',
  329. function(req, res) {
  330. // Check user is logged in
  331. //Authentication.checkLoggedIn(req.userId);
  332. const paramBoardId = req.params.boardId;
  333. Authentication.checkBoardAccess(req.userId, paramBoardId);
  334. // Check user has permission to add checklist to the card
  335. const board = Boards.findOne({
  336. _id: paramBoardId,
  337. });
  338. const addPermission = allowIsBoardMemberCommentOnly(req.userId, board);
  339. Authentication.checkAdminOrCondition(req.userId, addPermission);
  340. const paramCardId = req.params.cardId;
  341. const id = Checklists.insert({
  342. title: req.body.title,
  343. cardId: paramCardId,
  344. sort: 0,
  345. });
  346. if (id) {
  347. let items = req.body.items || [];
  348. if (_.isString(items)) {
  349. if (items === '') {
  350. items = [];
  351. } else {
  352. items = [items];
  353. }
  354. }
  355. items.forEach(function(item, idx) {
  356. ChecklistItems.insert({
  357. cardId: paramCardId,
  358. checklistId: id,
  359. title: item,
  360. sort: idx,
  361. });
  362. });
  363. JsonRoutes.sendResult(res, {
  364. code: 200,
  365. data: {
  366. _id: id,
  367. },
  368. });
  369. } else {
  370. JsonRoutes.sendResult(res, {
  371. code: 400,
  372. });
  373. }
  374. },
  375. );
  376. /**
  377. * @operation delete_checklist
  378. * @summary Delete a checklist
  379. *
  380. * @description The checklist will be removed, not put in the recycle bin.
  381. *
  382. * @param {string} boardId the board ID
  383. * @param {string} cardId the card ID
  384. * @param {string} checklistId the ID of the checklist to remove
  385. * @return_type {_id: string}
  386. */
  387. JsonRoutes.add(
  388. 'DELETE',
  389. '/api/boards/:boardId/cards/:cardId/checklists/:checklistId',
  390. function(req, res) {
  391. const paramBoardId = req.params.boardId;
  392. const paramChecklistId = req.params.checklistId;
  393. Authentication.checkBoardAccess(req.userId, paramBoardId);
  394. Checklists.remove({ _id: paramChecklistId });
  395. JsonRoutes.sendResult(res, {
  396. code: 200,
  397. data: {
  398. _id: paramChecklistId,
  399. },
  400. });
  401. },
  402. );
  403. }
  404. export default Checklists;