checklists.js 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  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. lastItem() {
  92. const allItems = this.items().fetch();
  93. const ret = allItems[allItems.length - 1];
  94. return ret;
  95. },
  96. finishedCount() {
  97. return ChecklistItems.find({
  98. checklistId: this._id,
  99. isFinished: true,
  100. }).count();
  101. },
  102. isFinished() {
  103. return 0 !== this.itemCount() && this.itemCount() === this.finishedCount();
  104. },
  105. checkAllItems() {
  106. const checkItems = ChecklistItems.find({ checklistId: this._id });
  107. checkItems.forEach(function(item) {
  108. item.check();
  109. });
  110. },
  111. uncheckAllItems() {
  112. const checkItems = ChecklistItems.find({ checklistId: this._id });
  113. checkItems.forEach(function(item) {
  114. item.uncheck();
  115. });
  116. },
  117. itemIndex(itemId) {
  118. const items = self.findOne({ _id: this._id }).items;
  119. return _.pluck(items, '_id').indexOf(itemId);
  120. },
  121. });
  122. Checklists.allow({
  123. insert(userId, doc) {
  124. return allowIsBoardMemberByCard(userId, Cards.findOne(doc.cardId));
  125. },
  126. update(userId, doc) {
  127. return allowIsBoardMemberByCard(userId, Cards.findOne(doc.cardId));
  128. },
  129. remove(userId, doc) {
  130. return allowIsBoardMemberByCard(userId, Cards.findOne(doc.cardId));
  131. },
  132. fetch: ['userId', 'cardId'],
  133. });
  134. Checklists.before.insert((userId, doc) => {
  135. doc.createdAt = new Date();
  136. if (!doc.userId) {
  137. doc.userId = userId;
  138. }
  139. });
  140. Checklists.mutations({
  141. setTitle(title) {
  142. return { $set: { title } };
  143. },
  144. /** move the checklist to another card
  145. * @param newCardId move the checklist to this cardId
  146. */
  147. move(newCardId) {
  148. // update every activity
  149. Activities.find(
  150. {checklistId: this._id}
  151. ).forEach(activity => {
  152. Activities.update(activity._id, {
  153. $set: {
  154. cardId: newCardId,
  155. },
  156. });
  157. });
  158. // update every checklist-item
  159. ChecklistItems.find(
  160. {checklistId: this._id}
  161. ).forEach(checklistItem => {
  162. ChecklistItems.update(checklistItem._id, {
  163. $set: {
  164. cardId: newCardId,
  165. },
  166. });
  167. });
  168. // update the checklist itself
  169. return {
  170. $set: {
  171. cardId: newCardId,
  172. },
  173. };
  174. },
  175. });
  176. if (Meteor.isServer) {
  177. Meteor.startup(() => {
  178. Checklists._collection._ensureIndex({ modifiedAt: -1 });
  179. Checklists._collection._ensureIndex({ cardId: 1, createdAt: 1 });
  180. });
  181. Checklists.after.insert((userId, doc) => {
  182. const card = Cards.findOne(doc.cardId);
  183. Activities.insert({
  184. userId,
  185. activityType: 'addChecklist',
  186. cardId: doc.cardId,
  187. boardId: card.boardId,
  188. checklistId: doc._id,
  189. checklistName: doc.title,
  190. listId: card.listId,
  191. swimlaneId: card.swimlaneId,
  192. });
  193. });
  194. Checklists.before.remove((userId, doc) => {
  195. const activities = Activities.find({ checklistId: doc._id });
  196. const card = Cards.findOne(doc.cardId);
  197. if (activities) {
  198. activities.forEach(activity => {
  199. Activities.remove(activity._id);
  200. });
  201. }
  202. Activities.insert({
  203. userId,
  204. activityType: 'removeChecklist',
  205. cardId: doc.cardId,
  206. boardId: Cards.findOne(doc.cardId).boardId,
  207. checklistId: doc._id,
  208. checklistName: doc.title,
  209. listId: card.listId,
  210. swimlaneId: card.swimlaneId,
  211. });
  212. });
  213. }
  214. if (Meteor.isServer) {
  215. /**
  216. * @operation get_all_checklists
  217. * @summary Get the list of checklists attached to a card
  218. *
  219. * @param {string} boardId the board ID
  220. * @param {string} cardId the card ID
  221. * @return_type [{_id: string,
  222. * title: string}]
  223. */
  224. JsonRoutes.add(
  225. 'GET',
  226. '/api/boards/:boardId/cards/:cardId/checklists',
  227. function(req, res) {
  228. Authentication.checkUserId(req.userId);
  229. const paramBoardId = req.params.boardId;
  230. const paramCardId = req.params.cardId;
  231. const checklists = Checklists.find({ cardId: paramCardId }).map(function(
  232. doc,
  233. ) {
  234. return {
  235. _id: doc._id,
  236. title: doc.title,
  237. };
  238. });
  239. if (checklists) {
  240. JsonRoutes.sendResult(res, {
  241. code: 200,
  242. data: checklists,
  243. });
  244. } else {
  245. JsonRoutes.sendResult(res, {
  246. code: 500,
  247. });
  248. }
  249. },
  250. );
  251. /**
  252. * @operation get_checklist
  253. * @summary Get a checklist
  254. *
  255. * @param {string} boardId the board ID
  256. * @param {string} cardId the card ID
  257. * @param {string} checklistId the ID of the checklist
  258. * @return_type {cardId: string,
  259. * title: string,
  260. * finishedAt: string,
  261. * createdAt: string,
  262. * sort: number,
  263. * items: [{_id: string,
  264. * title: string,
  265. * isFinished: boolean}]}
  266. */
  267. JsonRoutes.add(
  268. 'GET',
  269. '/api/boards/:boardId/cards/:cardId/checklists/:checklistId',
  270. function(req, res) {
  271. Authentication.checkUserId(req.userId);
  272. const paramBoardId = req.params.boardId;
  273. const paramChecklistId = req.params.checklistId;
  274. const paramCardId = req.params.cardId;
  275. const checklist = Checklists.findOne({
  276. _id: paramChecklistId,
  277. cardId: paramCardId,
  278. });
  279. if (checklist) {
  280. checklist.items = ChecklistItems.find({
  281. checklistId: checklist._id,
  282. }).map(function(doc) {
  283. return {
  284. _id: doc._id,
  285. title: doc.title,
  286. isFinished: doc.isFinished,
  287. };
  288. });
  289. JsonRoutes.sendResult(res, {
  290. code: 200,
  291. data: checklist,
  292. });
  293. } else {
  294. JsonRoutes.sendResult(res, {
  295. code: 500,
  296. });
  297. }
  298. },
  299. );
  300. /**
  301. * @operation new_checklist
  302. * @summary create a new checklist
  303. *
  304. * @param {string} boardId the board ID
  305. * @param {string} cardId the card ID
  306. * @param {string} title the title of the new checklist
  307. * @param {string} [items] the list of items on the new checklist
  308. * @return_type {_id: string}
  309. */
  310. JsonRoutes.add(
  311. 'POST',
  312. '/api/boards/:boardId/cards/:cardId/checklists',
  313. function(req, res) {
  314. Authentication.checkUserId(req.userId);
  315. // Check user is logged in
  316. //Authentication.checkLoggedIn(req.userId);
  317. const paramBoardId = req.params.boardId;
  318. // Check user has permission to add checklist to the card
  319. const board = Boards.findOne({
  320. _id: paramBoardId,
  321. });
  322. const addPermission = allowIsBoardMemberCommentOnly(req.userId, board);
  323. Authentication.checkAdminOrCondition(req.userId, addPermission);
  324. const paramCardId = req.params.cardId;
  325. const id = Checklists.insert({
  326. title: req.body.title,
  327. cardId: paramCardId,
  328. sort: 0,
  329. });
  330. if (id) {
  331. let items = req.body.items || [];
  332. if (_.isString(items)) {
  333. if (items === '') {
  334. items = [];
  335. } else {
  336. items = [items];
  337. }
  338. }
  339. items.forEach(function(item, idx) {
  340. ChecklistItems.insert({
  341. cardId: paramCardId,
  342. checklistId: id,
  343. title: item,
  344. sort: idx,
  345. });
  346. });
  347. JsonRoutes.sendResult(res, {
  348. code: 200,
  349. data: {
  350. _id: id,
  351. },
  352. });
  353. } else {
  354. JsonRoutes.sendResult(res, {
  355. code: 400,
  356. });
  357. }
  358. },
  359. );
  360. /**
  361. * @operation delete_checklist
  362. * @summary Delete a checklist
  363. *
  364. * @description The checklist will be removed, not put in the recycle bin.
  365. *
  366. * @param {string} boardId the board ID
  367. * @param {string} cardId the card ID
  368. * @param {string} checklistId the ID of the checklist to remove
  369. * @return_type {_id: string}
  370. */
  371. JsonRoutes.add(
  372. 'DELETE',
  373. '/api/boards/:boardId/cards/:cardId/checklists/:checklistId',
  374. function(req, res) {
  375. Authentication.checkUserId(req.userId);
  376. const paramBoardId = req.params.boardId;
  377. const paramChecklistId = req.params.checklistId;
  378. Checklists.remove({ _id: paramChecklistId });
  379. JsonRoutes.sendResult(res, {
  380. code: 200,
  381. data: {
  382. _id: paramChecklistId,
  383. },
  384. });
  385. },
  386. );
  387. }
  388. export default Checklists;