| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592 | 
							- import { ReactiveCache } from '/imports/reactiveCache';
 
- import { ALLOWED_COLORS } from '/config/const';
 
- Swimlanes = new Mongo.Collection('swimlanes');
 
- /**
 
-  * A swimlane is an line in the kaban board.
 
-  */
 
- Swimlanes.attachSchema(
 
-   new SimpleSchema({
 
-     title: {
 
-       /**
 
-        * the title of the swimlane
 
-        */
 
-       type: String,
 
-     },
 
-     archived: {
 
-       /**
 
-        * is the swimlane archived?
 
-        */
 
-       type: Boolean,
 
-       // eslint-disable-next-line consistent-return
 
-       autoValue() {
 
-         if (this.isInsert && !this.isSet) {
 
-           return false;
 
-         }
 
-       },
 
-     },
 
-     archivedAt: {
 
-       /**
 
-        * latest archiving date of the swimlane
 
-        */
 
-       type: Date,
 
-       optional: true,
 
-     },
 
-     boardId: {
 
-       /**
 
-        * the ID of the board the swimlane is attached to
 
-        */
 
-       type: String,
 
-     },
 
-     createdAt: {
 
-       /**
 
-        * creation date of the swimlane
 
-        */
 
-       type: Date,
 
-       // eslint-disable-next-line consistent-return
 
-       autoValue() {
 
-         if (this.isInsert) {
 
-           return new Date();
 
-         } else if (this.isUpsert) {
 
-           return { $setOnInsert: new Date() };
 
-         } else {
 
-           this.unset();
 
-         }
 
-       },
 
-     },
 
-     sort: {
 
-       /**
 
-        * the sort value of the swimlane
 
-        */
 
-       type: Number,
 
-       decimal: true,
 
-       // XXX We should probably provide a default
 
-       optional: true,
 
-     },
 
-     color: {
 
-       /**
 
-        * the color of the swimlane
 
-        */
 
-       type: String,
 
-       optional: true,
 
-       // silver is the default, so it is left out
 
-       allowedValues: ALLOWED_COLORS,
 
-     },
 
-     updatedAt: {
 
-       /**
 
-        * when was the swimlane last edited
 
-        */
 
-       type: Date,
 
-       optional: true,
 
-       // eslint-disable-next-line consistent-return
 
-       autoValue() {
 
-         if (this.isUpdate || this.isUpsert || this.isInsert) {
 
-           return new Date();
 
-         } else {
 
-           this.unset();
 
-         }
 
-       },
 
-     },
 
-     modifiedAt: {
 
-       type: Date,
 
-       denyUpdate: false,
 
-       // eslint-disable-next-line consistent-return
 
-       autoValue() {
 
-         if (this.isInsert || this.isUpsert || this.isUpdate) {
 
-           return new Date();
 
-         } else {
 
-           this.unset();
 
-         }
 
-       },
 
-     },
 
-     type: {
 
-       /**
 
-        * The type of swimlane
 
-        */
 
-       type: String,
 
-       defaultValue: 'swimlane',
 
-     },
 
-   }),
 
- );
 
- Swimlanes.allow({
 
-   insert(userId, doc) {
 
-     return allowIsBoardMemberCommentOnly(userId, ReactiveCache.getBoard(doc.boardId));
 
-   },
 
-   update(userId, doc) {
 
-     return allowIsBoardMemberCommentOnly(userId, ReactiveCache.getBoard(doc.boardId));
 
-   },
 
-   remove(userId, doc) {
 
-     return allowIsBoardMemberCommentOnly(userId, ReactiveCache.getBoard(doc.boardId));
 
-   },
 
-   fetch: ['boardId'],
 
- });
 
- Swimlanes.helpers({
 
-   copy(boardId) {
 
-     const oldId = this._id;
 
-     const oldBoardId = this.boardId;
 
-     this.boardId = boardId;
 
-     delete this._id;
 
-     const _id = Swimlanes.insert(this);
 
-     const query = {
 
-       swimlaneId: { $in: [oldId, ''] },
 
-       archived: false,
 
-     };
 
-     if (oldBoardId) {
 
-       query.boardId = oldBoardId;
 
-     }
 
-     // Copy all lists in swimlane
 
-     ReactiveCache.getLists(query).forEach(list => {
 
-       list.type = 'list';
 
-       list.swimlaneId = oldId;
 
-       list.boardId = boardId;
 
-       list.copy(boardId, _id);
 
-     });
 
-   },
 
-   move(toBoardId) {
 
-     this.lists().forEach(list => {
 
-       const toList = ReactiveCache.getList({
 
-         boardId: toBoardId,
 
-         title: list.title,
 
-         archived: false,
 
-       });
 
-       let toListId;
 
-       if (toList) {
 
-         toListId = toList._id;
 
-       } else {
 
-         toListId = Lists.insert({
 
-           title: list.title,
 
-           boardId: toBoardId,
 
-           type: list.type,
 
-           archived: false,
 
-           wipLimit: list.wipLimit,
 
-         });
 
-       }
 
-       ReactiveCache.getCards({
 
-         listId: list._id,
 
-         swimlaneId: this._id,
 
-       }).forEach(card => {
 
-         card.move(toBoardId, this._id, toListId);
 
-       });
 
-     });
 
-     Swimlanes.update(this._id, {
 
-       $set: {
 
-         boardId: toBoardId,
 
-       },
 
-     });
 
-     // make sure there is a default swimlane
 
-     this.board().getDefaultSwimline();
 
-   },
 
-   cards() {
 
-     const ret = ReactiveCache.getCards(
 
-       Filter.mongoSelector({
 
-         swimlaneId: this._id,
 
-         archived: false,
 
-       }),
 
-       { sort: ['sort'] },
 
-     );
 
-     return ret;
 
-   },
 
-   lists() {
 
-     return this.draggableLists();
 
-   },
 
-   newestLists() {
 
-     // sorted lists from newest to the oldest, by its creation date or its cards' last modification date
 
-     return ReactiveCache.getLists(
 
-       {
 
-         boardId: this.boardId,
 
-         swimlaneId: { $in: [this._id, ''] },
 
-         archived: false,
 
-       },
 
-       { sort: { modifiedAt: -1 } },
 
-     );
 
-   },
 
-   draggableLists() {
 
-     return ReactiveCache.getLists(
 
-       {
 
-         boardId: this.boardId,
 
-         swimlaneId: { $in: [this._id, ''] },
 
-         //archived: false,
 
-       },
 
-       { sort: ['sort'] },
 
-     );
 
-   },
 
-   myLists() {
 
-     return ReactiveCache.getLists({ swimlaneId: this._id });
 
-   },
 
-   allCards() {
 
-     const ret = ReactiveCache.getCards({ swimlaneId: this._id });
 
-     return ret;
 
-   },
 
-   board() {
 
-     return ReactiveCache.getBoard(this.boardId);
 
-   },
 
-   colorClass() {
 
-     if (this.color) return `swimlane-${this.color}`;
 
-     return '';
 
-   },
 
-   isTemplateSwimlane() {
 
-     return this.type === 'template-swimlane';
 
-   },
 
-   isTemplateContainer() {
 
-     return this.type === 'template-container';
 
-   },
 
-   isListTemplatesSwimlane() {
 
-     const user = ReactiveCache.getCurrentUser();
 
-     return (user.profile || {}).listTemplatesSwimlaneId === this._id;
 
-   },
 
-   isCardTemplatesSwimlane() {
 
-     const user = ReactiveCache.getCurrentUser();
 
-     return (user.profile || {}).cardTemplatesSwimlaneId === this._id;
 
-   },
 
-   isBoardTemplatesSwimlane() {
 
-     const user = ReactiveCache.getCurrentUser();
 
-     return (user.profile || {}).boardTemplatesSwimlaneId === this._id;
 
-   },
 
-   remove() {
 
-     Swimlanes.remove({ _id: this._id });
 
-   },
 
- });
 
- Swimlanes.mutations({
 
-   rename(title) {
 
-     return { $set: { title } };
 
-   },
 
-   archive() {
 
-     if (this.isTemplateSwimlane()) {
 
-       this.myLists().forEach(list => {
 
-         return list.archive();
 
-       });
 
-     }
 
-     return { $set: { archived: true, archivedAt: new Date() } };
 
-   },
 
-   restore() {
 
-     if (this.isTemplateSwimlane()) {
 
-       this.myLists().forEach(list => {
 
-         return list.restore();
 
-       });
 
-     }
 
-     return { $set: { archived: false } };
 
-   },
 
-   setColor(newColor) {
 
-     if (newColor === 'silver') {
 
-       newColor = null;
 
-     }
 
-     return {
 
-       $set: {
 
-         color: newColor,
 
-       },
 
-     };
 
-   },
 
- });
 
- Swimlanes.userArchivedSwimlanes = userId => {
 
-   return ReactiveCache.getSwimlanes({
 
-     boardId: { $in: Boards.userBoardIds(userId, null) },
 
-     archived: true,
 
-   })
 
- };
 
- Swimlanes.userArchivedSwimlaneIds = () => {
 
-   return Swimlanes.userArchivedSwimlanes().map(swim => { return swim._id; });
 
- };
 
- Swimlanes.archivedSwimlanes = () => {
 
-   return ReactiveCache.getSwimlanes({ archived: true });
 
- };
 
- Swimlanes.archivedSwimlaneIds = () => {
 
-   return Swimlanes.archivedSwimlanes().map(swim => {
 
-     return swim._id;
 
-   });
 
- };
 
- Swimlanes.hookOptions.after.update = { fetchPrevious: false };
 
- if (Meteor.isServer) {
 
-   Meteor.startup(() => {
 
-     Swimlanes._collection.createIndex({ modifiedAt: -1 });
 
-     Swimlanes._collection.createIndex({ boardId: 1 });
 
-   });
 
-   Swimlanes.after.insert((userId, doc) => {
 
-     Activities.insert({
 
-       userId,
 
-       type: 'swimlane',
 
-       activityType: 'createSwimlane',
 
-       boardId: doc.boardId,
 
-       swimlaneId: doc._id,
 
-     });
 
-   });
 
-   Swimlanes.before.remove(function(userId, doc) {
 
-     const lists = ReactiveCache.getLists(
 
-       {
 
-         boardId: doc.boardId,
 
-         swimlaneId: { $in: [doc._id, ''] },
 
-         archived: false,
 
-       },
 
-       { sort: ['sort'] },
 
-     );
 
-     if (lists.length < 2) {
 
-       lists.forEach(list => {
 
-         list.remove();
 
-       });
 
-     } else {
 
-       Cards.remove({ swimlaneId: doc._id });
 
-     }
 
-     Activities.insert({
 
-       userId,
 
-       type: 'swimlane',
 
-       activityType: 'removeSwimlane',
 
-       boardId: doc.boardId,
 
-       swimlaneId: doc._id,
 
-       title: doc.title,
 
-     });
 
-   });
 
-   Swimlanes.after.update((userId, doc, fieldNames) => {
 
-     if (fieldNames.includes('title')) {
 
-       Activities.insert({
 
-         userId,
 
-         type: 'swimlane',
 
-         activityType: 'changedSwimlaneTitle',
 
-         listId: doc._id,
 
-         boardId: doc.boardId,
 
-         // this preserves the name so that the activity can be useful after the
 
-         // list is deleted
 
-         title: doc.title,
 
-       });
 
-     } else if (doc.archived)  {
 
-       Activities.insert({
 
-         userId,
 
-         type: 'swimlane',
 
-         activityType: 'archivedSwimlane',
 
-         listId: doc._id,
 
-         boardId: doc.boardId,
 
-         // this preserves the name so that the activity can be useful after the
 
-         // list is deleted
 
-         title: doc.title,
 
-       });
 
-     } else if (fieldNames.includes('archived'))  {
 
-       Activities.insert({
 
-         userId,
 
-         type: 'swimlane',
 
-         activityType: 'restoredSwimlane',
 
-         listId: doc._id,
 
-         boardId: doc.boardId,
 
-         // this preserves the name so that the activity can be useful after the
 
-         // list is deleted
 
-         title: doc.title,
 
-       });
 
-     }
 
-   });
 
- }
 
- //SWIMLANE REST API
 
- if (Meteor.isServer) {
 
-   /**
 
-    * @operation get_all_swimlanes
 
-    *
 
-    * @summary Get the list of swimlanes attached to a board
 
-    *
 
-    * @param {string} boardId the ID of the board
 
-    * @return_type [{_id: string,
 
-    *                title: string}]
 
-    */
 
-   JsonRoutes.add('GET', '/api/boards/:boardId/swimlanes', function(req, res) {
 
-     try {
 
-       const paramBoardId = req.params.boardId;
 
-       Authentication.checkBoardAccess(req.userId, paramBoardId);
 
-       JsonRoutes.sendResult(res, {
 
-         code: 200,
 
-         data: ReactiveCache.getSwimlanes({ boardId: paramBoardId, archived: false }).map(
 
-           function(doc) {
 
-             return {
 
-               _id: doc._id,
 
-               title: doc.title,
 
-             };
 
-           },
 
-         ),
 
-       });
 
-     } catch (error) {
 
-       JsonRoutes.sendResult(res, {
 
-         code: 200,
 
-         data: error,
 
-       });
 
-     }
 
-   });
 
-   /**
 
-    * @operation get_swimlane
 
-    *
 
-    * @summary Get a swimlane
 
-    *
 
-    * @param {string} boardId the ID of the board
 
-    * @param {string} swimlaneId the ID of the swimlane
 
-    * @return_type Swimlanes
 
-    */
 
-   JsonRoutes.add('GET', '/api/boards/:boardId/swimlanes/:swimlaneId', function(
 
-     req,
 
-     res,
 
-   ) {
 
-     try {
 
-       const paramBoardId = req.params.boardId;
 
-       const paramSwimlaneId = req.params.swimlaneId;
 
-       Authentication.checkBoardAccess(req.userId, paramBoardId);
 
-       JsonRoutes.sendResult(res, {
 
-         code: 200,
 
-         data: ReactiveCache.getSwimlane({
 
-           _id: paramSwimlaneId,
 
-           boardId: paramBoardId,
 
-           archived: false,
 
-         }),
 
-       });
 
-     } catch (error) {
 
-       JsonRoutes.sendResult(res, {
 
-         code: 200,
 
-         data: error,
 
-       });
 
-     }
 
-   });
 
-   /**
 
-    * @operation new_swimlane
 
-    *
 
-    * @summary Add a swimlane to a board
 
-    *
 
-    * @param {string} boardId the ID of the board
 
-    * @param {string} title the new title of the swimlane
 
-    * @return_type {_id: string}
 
-    */
 
-   JsonRoutes.add('POST', '/api/boards/:boardId/swimlanes', function(req, res) {
 
-     try {
 
-       const paramBoardId = req.params.boardId;
 
-       Authentication.checkBoardAccess(req.userId, paramBoardId);
 
-       const board = ReactiveCache.getBoard(paramBoardId);
 
-       const id = Swimlanes.insert({
 
-         title: req.body.title,
 
-         boardId: paramBoardId,
 
-         sort: board.swimlanes().length,
 
-       });
 
-       JsonRoutes.sendResult(res, {
 
-         code: 200,
 
-         data: {
 
-           _id: id,
 
-         },
 
-       });
 
-     } catch (error) {
 
-       JsonRoutes.sendResult(res, {
 
-         code: 200,
 
-         data: error,
 
-       });
 
-     }
 
-   });
 
-   /**
 
-    * @operation edit_swimlane
 
-    *
 
-    * @summary Edit the title of a swimlane
 
-    *
 
-    * @param {string} boardId the ID of the board
 
-    * @param {string} swimlaneId the ID of the swimlane to edit
 
-    * @param {string} title the new title of the swimlane
 
-    * @return_type {_id: string}
 
-    */
 
-   JsonRoutes.add('PUT', '/api/boards/:boardId/swimlanes/:swimlaneId', function(req, res) {
 
-     try {
 
-       const paramBoardId = req.params.boardId;
 
-       const paramSwimlaneId = req.params.swimlaneId;
 
-       Authentication.checkBoardAccess(req.userId, paramBoardId);
 
-       const board = ReactiveCache.getBoard(paramBoardId);
 
-       const swimlane = ReactiveCache.getSwimlane({
 
-         _id: paramSwimlaneId,
 
-         boardId: paramBoardId,
 
-       });
 
-       if (!swimlane) {
 
-         throw new Meteor.Error('not-found', 'Swimlane not found');
 
-       }
 
-       Swimlanes.direct.update(
 
-         { _id: paramSwimlaneId },
 
-         { $set: { title: req.body.title } }
 
-       );
 
-       JsonRoutes.sendResult(res, {
 
-         code: 200,
 
-         data: {
 
-           _id: paramSwimlaneId,
 
-         },
 
-       });
 
-     } catch (error) {
 
-       JsonRoutes.sendResult(res, {
 
-         code: 200,
 
-         data: error,
 
-       });
 
-     }
 
-   });
 
-   /**
 
-    * @operation delete_swimlane
 
-    *
 
-    * @summary Delete a swimlane
 
-    *
 
-    * @description The swimlane will be deleted, not moved to the recycle bin
 
-    *
 
-    * @param {string} boardId the ID of the board
 
-    * @param {string} swimlaneId the ID of the swimlane
 
-    * @return_type {_id: string}
 
-    */
 
-   JsonRoutes.add(
 
-     'DELETE',
 
-     '/api/boards/:boardId/swimlanes/:swimlaneId',
 
-     function(req, res) {
 
-       try {
 
-         const paramBoardId = req.params.boardId;
 
-         const paramSwimlaneId = req.params.swimlaneId;
 
-         Authentication.checkBoardAccess(req.userId, paramBoardId);
 
-         Swimlanes.remove({ _id: paramSwimlaneId, boardId: paramBoardId });
 
-         JsonRoutes.sendResult(res, {
 
-           code: 200,
 
-           data: {
 
-             _id: paramSwimlaneId,
 
-           },
 
-         });
 
-       } catch (error) {
 
-         JsonRoutes.sendResult(res, {
 
-           code: 200,
 
-           data: error,
 
-         });
 
-       }
 
-     },
 
-   );
 
- }
 
- export default Swimlanes;
 
 
  |