| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591 | import { ALLOWED_COLORS } from '/config/const';Lists = new Mongo.Collection('lists');/** * A list (column) in the Wekan board. */Lists.attachSchema(  new SimpleSchema({    title: {      /**       * the title of the list       */      type: String,    },    starred: {      /**       * if a list is stared       * then we put it on the top       */      type: Boolean,      optional: true,      defaultValue: false,    },    archived: {      /**       * is the list archived       */      type: Boolean,      // eslint-disable-next-line consistent-return      autoValue() {        if (this.isInsert && !this.isSet) {          return false;        }      },    },    archivedAt: {      /**       * latest archiving date       */      type: Date,      optional: true,    },    boardId: {      /**       * the board associated to this list       */      type: String,    },    swimlaneId: {      /**       * the swimlane associated to this list. Used for templates       */      type: String,      defaultValue: '',    },    createdAt: {      /**       * creation date       */      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: {      /**       * is the list sorted       */      type: Number,      decimal: true,      // XXX We should probably provide a default      optional: true,    },    updatedAt: {      /**       * last update of the list       */      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() {        // this is redundant with updatedAt        /*if (this.isInsert || this.isUpsert || this.isUpdate) {          return new Date();        } else {          this.unset();        }*/        if (!this.isSet) {          return new Date();        }      },    },    wipLimit: {      /**       * WIP object, see below       */      type: Object,      optional: true,    },    'wipLimit.value': {      /**       * value of the WIP       */      type: Number,      decimal: false,      defaultValue: 1,    },    'wipLimit.enabled': {      /**       * is the WIP enabled       */      type: Boolean,      defaultValue: false,    },    'wipLimit.soft': {      /**       * is the WIP a soft or hard requirement       */      type: Boolean,      defaultValue: false,    },    color: {      /**       * the color of the list       */      type: String,      optional: true,      // silver is the default, so it is left out      allowedValues: ALLOWED_COLORS,    },    type: {      /**       * The type of list       */      type: String,      defaultValue: 'list',    },  }),);Lists.allow({  insert(userId, doc) {    return allowIsBoardMemberCommentOnly(userId, Boards.findOne(doc.boardId));  },  update(userId, doc) {    return allowIsBoardMemberCommentOnly(userId, Boards.findOne(doc.boardId));  },  remove(userId, doc) {    return allowIsBoardMemberCommentOnly(userId, Boards.findOne(doc.boardId));  },  fetch: ['boardId'],});Lists.helpers({  copy(boardId, swimlaneId) {    const oldId = this._id;    const oldSwimlaneId = this.swimlaneId || null;    this.boardId = boardId;    this.swimlaneId = swimlaneId;    let _id = null;    const existingListWithSameName = Lists.findOne({      boardId,      title: this.title,      archived: false,    });    if (existingListWithSameName) {      _id = existingListWithSameName._id;    } else {      delete this._id;      delete this.swimlaneId;      _id = Lists.insert(this);    }    // Copy all cards in list    Cards.find({      swimlaneId: oldSwimlaneId,      listId: oldId,      archived: false,    }).forEach(card => {      card.copy(boardId, swimlaneId, _id);    });  },  move(boardId, swimlaneId) {    const boardList = Lists.findOne({      boardId,      title: this.title,      archived: false,    });    let listId;    if (boardList) {      listId = boardList._id;      this.cards().forEach(card => {        card.move(boardId, this._id, boardList._id);      });    } else {      console.log('list.title:', this.title);      console.log('boardList:', boardList);      listId = Lists.insert({        title: this.title,        boardId,        type: this.type,        archived: false,        wipLimit: this.wipLimit,      });    }    this.cards(swimlaneId).forEach(card => {      card.move(boardId, swimlaneId, listId);    });  },  cards(swimlaneId) {    const selector = {      listId: this._id,      archived: false,    };    if (swimlaneId) selector.swimlaneId = swimlaneId;    return Cards.find(Filter.mongoSelector(selector), { sort: ['sort'] });  },  cardsUnfiltered(swimlaneId) {    const selector = {      listId: this._id,      archived: false,    };    if (swimlaneId) selector.swimlaneId = swimlaneId;    return Cards.find(selector, { sort: ['sort'] });  },  allCards() {    return Cards.find({ listId: this._id });  },  board() {    return Boards.findOne(this.boardId);  },  getWipLimit(option) {    const list = Lists.findOne({ _id: this._id });    if (!list.wipLimit) {      // Necessary check to avoid exceptions for the case where the doc doesn't have the wipLimit field yet set      return 0;    } else if (!option) {      return list.wipLimit;    } else {      return list.wipLimit[option] ? list.wipLimit[option] : 0; // Necessary check to avoid exceptions for the case where the doc doesn't have the wipLimit field yet set    }  },  colorClass() {    if (this.color) return `list-header-${this.color}`;    return '';  },  isTemplateList() {    return this.type === 'template-list';  },  isStarred() {    return this.starred === true;  },  absoluteUrl() {    const card = Cards.findOne({ listId: this._id });    return card && card.absoluteUrl();  },  originRelativeUrl() {    const card = Cards.findOne({ listId: this._id });    return card && card.originRelativeUrl();  },  remove() {    Lists.remove({ _id: this._id });  },});Lists.mutations({  rename(title) {    return { $set: { title } };  },  star(enable = true) {    return { $set: { starred: !!enable } };  },  archive() {    if (this.isTemplateList()) {      this.cards().forEach(card => {        return card.archive();      });    }    return { $set: { archived: true, archivedAt: new Date() } };  },  restore() {    if (this.isTemplateList()) {      this.allCards().forEach(card => {        return card.restore();      });    }    return { $set: { archived: false } };  },  toggleSoftLimit(toggle) {    return { $set: { 'wipLimit.soft': toggle } };  },  toggleWipLimit(toggle) {    return { $set: { 'wipLimit.enabled': toggle } };  },  setWipLimit(limit) {    return { $set: { 'wipLimit.value': limit } };  },  setColor(newColor) {    if (newColor === 'silver') {      newColor = null;    }    return {      $set: {        color: newColor,      },    };  },});Lists.archivedLists = () => {  return Lists.find({ archived: true });};Lists.archivedListIds = () => {  return Lists.archivedLists().map(list => {    return list._id;  });};Meteor.methods({  applyWipLimit(listId, limit) {    check(listId, String);    check(limit, Number);    if (limit === 0) {      limit = 1;    }    Lists.findOne({ _id: listId }).setWipLimit(limit);  },  enableWipLimit(listId) {    check(listId, String);    const list = Lists.findOne({ _id: listId });    if (list.getWipLimit('value') === 0) {      list.setWipLimit(1);    }    list.toggleWipLimit(!list.getWipLimit('enabled'));  },  enableSoftLimit(listId) {    check(listId, String);    const list = Lists.findOne({ _id: listId });    list.toggleSoftLimit(!list.getWipLimit('soft'));  },  myLists() {    // my lists    return _.uniq(      Lists.find(        {          boardId: { $in: Boards.userBoardIds(this.userId) },          archived: false,        },        {          fields: { title: 1 },        },      )        .fetch()        .map(list => {          return list.title;        }),    ).sort();  },});Lists.hookOptions.after.update = { fetchPrevious: false };if (Meteor.isServer) {  Meteor.startup(() => {    Lists._collection._ensureIndex({ modifiedAt: -1 });    Lists._collection._ensureIndex({ boardId: 1 });    Lists._collection._ensureIndex({ archivedAt: -1 });  });  Lists.after.insert((userId, doc) => {    Activities.insert({      userId,      type: 'list',      activityType: 'createList',      boardId: doc.boardId,      listId: doc._id,      // this preserves the name so that the activity can be useful after the      // list is deleted      title: doc.title,    });  });  Lists.before.remove((userId, doc) => {    const cards = Cards.find({ listId: doc._id });    if (cards) {      cards.forEach(card => {        Cards.remove(card._id);      });    }    Activities.insert({      userId,      type: 'list',      activityType: 'removeList',      boardId: doc.boardId,      listId: doc._id,      title: doc.title,    });  });  Lists.after.update((userId, doc) => {    if (doc.archived) {      Activities.insert({        userId,        type: 'list',        activityType: 'archivedList',        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,      });    }  });}//LISTS REST APIif (Meteor.isServer) {  /**   * @operation get_all_lists   * @summary Get the list of Lists attached to a board   *   * @param {string} boardId the board ID   * @return_type [{_id: string,   *           title: string}]   */  JsonRoutes.add('GET', '/api/boards/:boardId/lists', function(req, res) {    try {      const paramBoardId = req.params.boardId;      Authentication.checkBoardAccess(req.userId, paramBoardId);      JsonRoutes.sendResult(res, {        code: 200,        data: Lists.find({ 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_list   * @summary Get a List attached to a board   *   * @param {string} boardId the board ID   * @param {string} listId the List ID   * @return_type Lists   */  JsonRoutes.add('GET', '/api/boards/:boardId/lists/:listId', function(    req,    res,  ) {    try {      const paramBoardId = req.params.boardId;      const paramListId = req.params.listId;      Authentication.checkBoardAccess(req.userId, paramBoardId);      JsonRoutes.sendResult(res, {        code: 200,        data: Lists.findOne({          _id: paramListId,          boardId: paramBoardId,          archived: false,        }),      });    } catch (error) {      JsonRoutes.sendResult(res, {        code: 200,        data: error,      });    }  });  /**   * @operation new_list   * @summary Add a List to a board   *   * @param {string} boardId the board ID   * @param {string} title the title of the List   * @return_type {_id: string}   */  JsonRoutes.add('POST', '/api/boards/:boardId/lists', function(req, res) {    try {      const paramBoardId = req.params.boardId;      Authentication.checkBoardAccess(req.userId, paramBoardId);      const board = Boards.findOne(paramBoardId);      const id = Lists.insert({        title: req.body.title,        boardId: paramBoardId,        sort: board.lists().count(),      });      JsonRoutes.sendResult(res, {        code: 200,        data: {          _id: id,        },      });    } catch (error) {      JsonRoutes.sendResult(res, {        code: 200,        data: error,      });    }  });  /**   * @operation delete_list   * @summary Delete a List   *   * @description This **deletes** a list from a board.   * The list is not put in the recycle bin.   *   * @param {string} boardId the board ID   * @param {string} listId the ID of the list to remove   * @return_type {_id: string}   */  JsonRoutes.add('DELETE', '/api/boards/:boardId/lists/:listId', function(    req,    res,  ) {    try {      const paramBoardId = req.params.boardId;      Authentication.checkBoardAccess(req.userId, paramBoardId);      const paramListId = req.params.listId;      Lists.remove({ _id: paramListId, boardId: paramBoardId });      JsonRoutes.sendResult(res, {        code: 200,        data: {          _id: paramListId,        },      });    } catch (error) {      JsonRoutes.sendResult(res, {        code: 200,        data: error,      });    }  });}export default Lists;
 |