| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890 | import { ReactiveCache } from '/imports/reactiveCache';import { ALLOWED_COLORS } from '/config/const';import PositionHistory from './positionHistory';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. Optional for backward compatibility       */      type: String,      optional: true,      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      allowedValues: ALLOWED_COLORS,    },    type: {      /**       * The type of list       */      type: String,      defaultValue: 'list',    },    collapsed: {      /**       * is the list collapsed       */      type: Boolean,      defaultValue: false,    },  }),);Lists.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'],});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 = ReactiveCache.getList({      boardId,      title: this.title,      archived: false,    });    if (existingListWithSameName) {      _id = existingListWithSameName._id;    } else {      delete this._id;      this.swimlaneId = swimlaneId; // Set the target swimlane for the copied list      _id = Lists.insert(this);    }    // Copy all cards in list    ReactiveCache.getCards({      swimlaneId: oldSwimlaneId,      listId: oldId,      archived: false,    }).forEach(card => {      card.copy(boardId, swimlaneId, _id);    });  },  move(boardId, swimlaneId) {    const boardList = ReactiveCache.getList({      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,        swimlaneId: swimlaneId, // Set the target swimlane for the moved list      });    }    this.cards(swimlaneId).forEach(card => {      card.move(boardId, swimlaneId, listId);    });  },  cards(swimlaneId) {    const selector = {      listId: this._id,      archived: false,    };    if (swimlaneId) selector.swimlaneId = swimlaneId;    const ret = ReactiveCache.getCards(Filter.mongoSelector(selector), { sort: ['sort'] });    return ret;  },  cardsUnfiltered(swimlaneId) {    const selector = {      listId: this._id,      archived: false,    };    if (swimlaneId) selector.swimlaneId = swimlaneId;    const ret = ReactiveCache.getCards(selector, { sort: ['sort'] });    return ret;  },  allCards() {    const ret = ReactiveCache.getCards({ listId: this._id });    return ret;  },  board() {    return ReactiveCache.getBoard(this.boardId);  },  getWipLimit(option) {    const list = ReactiveCache.getList(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;  },  isCollapsed() {    return this.collapsed === true;  },  absoluteUrl() {    const card = ReactiveCache.getCard({ listId: this._id });    return card && card.absoluteUrl();  },  originRelativeUrl() {    const card = ReactiveCache.getCard({ listId: this._id });    return card && card.originRelativeUrl();  },  remove() {    Lists.remove({ _id: this._id });  },});Lists.mutations({  rename(title) {    // Basic client-side validation - server will handle full sanitization    if (typeof title === 'string') {      // Basic length check to prevent abuse      const sanitizedTitle = title.length > 1000 ? title.substring(0, 1000) : title;      return { $set: { title: sanitizedTitle } };    }    return { $set: { title } };  },  star(enable = true) {    return { $set: { starred: !!enable } };  },  collapse(enable = true) {    return { $set: { collapsed: !!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) {    return {      $set: {        color: newColor,      },    };  },});Lists.userArchivedLists = userId => {  return ReactiveCache.getLists({    boardId: { $in: Boards.userBoardIds(userId, null) },    archived: true,  })};Lists.userArchivedListIds = () => {  return Lists.userArchivedLists().map(list => { return list._id; });};Lists.archivedLists = () => {  return ReactiveCache.getLists({ 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;    }    ReactiveCache.getList(listId).setWipLimit(limit);  },  enableWipLimit(listId) {    check(listId, String);    const list = ReactiveCache.getList(listId);    if (list.getWipLimit('value') === 0) {      list.setWipLimit(1);    }    list.toggleWipLimit(!list.getWipLimit('enabled'));  },  enableSoftLimit(listId) {    check(listId, String);    const list = ReactiveCache.getList(listId);    list.toggleSoftLimit(!list.getWipLimit('soft'));  },  myLists() {    // my lists    return _.uniq(      ReactiveCache.getLists(        {          boardId: { $in: Boards.userBoardIds(this.userId) },          archived: false,        },        {          fields: { title: 1 },        },      )        .map(list => {          return list.title;        }),    ).sort();  },});Lists.hookOptions.after.update = { fetchPrevious: false };if (Meteor.isServer) {  Meteor.startup(() => {    Lists._collection.createIndex({ modifiedAt: -1 });    Lists._collection.createIndex({ boardId: 1 });    Lists._collection.createIndex({ 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,    });    // Track original position for new lists    Meteor.setTimeout(() => {      const list = Lists.findOne(doc._id);      if (list) {        list.trackOriginalPosition();      }    }, 100);  });  Lists.before.remove((userId, doc) => {    const cards = ReactiveCache.getCards({ 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, fieldNames) => {    if (fieldNames.includes('title')) {      Activities.insert({        userId,        type: 'list',        activityType: 'changedListTitle',        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: '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,      });    } else if (fieldNames.includes('archived'))  {      Activities.insert({        userId,        type: 'list',        activityType: 'restoredList',        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: ReactiveCache.getLists({ 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: ReactiveCache.getList({          _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 = ReactiveCache.getBoard(paramBoardId);      const id = Lists.insert({        title: req.body.title,        boardId: paramBoardId,        sort: board.lists().length,        swimlaneId: req.body.swimlaneId || board.getDefaultSwimline()._id, // Use provided swimlaneId or default      });      JsonRoutes.sendResult(res, {        code: 200,        data: {          _id: id,        },      });    } catch (error) {      JsonRoutes.sendResult(res, {        code: 200,        data: error,      });    }  });  /**   * @operation edit_list   * @summary Edit a List   *   * @description This updates a list on a board.   * You can update the title, color, wipLimit, starred, and collapsed properties.   *   * @param {string} boardId the board ID   * @param {string} listId the ID of the list to update   * @param {string} [title] the new title of the list   * @param {string} [color] the new color of the list   * @param {Object} [wipLimit] the WIP limit configuration   * @param {boolean} [starred] whether the list is starred   * @param {boolean} [collapsed] whether the list is collapsed   * @return_type {_id: string}   */  JsonRoutes.add('PUT', '/api/boards/:boardId/lists/:listId', function(    req,    res,  ) {    try {      const paramBoardId = req.params.boardId;      const paramListId = req.params.listId;      let updated = false;      Authentication.checkBoardAccess(req.userId, paramBoardId);      const list = ReactiveCache.getList({        _id: paramListId,        boardId: paramBoardId,        archived: false,      });      if (!list) {        JsonRoutes.sendResult(res, {          code: 404,          data: { error: 'List not found' },        });        return;      }      // Update title if provided      if (req.body.title) {        // Basic client-side validation - server will handle full sanitization        const newTitle = req.body.title.length > 1000 ? req.body.title.substring(0, 1000) : req.body.title;        if (process.env.DEBUG === 'true' && newTitle !== req.body.title) {          console.warn('Sanitized list title input:', req.body.title, '->', newTitle);        }        Lists.direct.update(          {            _id: paramListId,            boardId: paramBoardId,            archived: false,          },          {            $set: {              title: newTitle,            },          },        );        updated = true;      }      // Update color if provided      if (req.body.color) {        const newColor = req.body.color;        Lists.direct.update(          {            _id: paramListId,            boardId: paramBoardId,            archived: false,          },          {            $set: {              color: newColor,            },          },        );        updated = true;      }      // Update starred status if provided      if (req.body.hasOwnProperty('starred')) {        const newStarred = req.body.starred;        Lists.direct.update(          {            _id: paramListId,            boardId: paramBoardId,            archived: false,          },          {            $set: {              starred: newStarred,            },          },        );        updated = true;      }      // Update collapsed status if provided      if (req.body.hasOwnProperty('collapsed')) {        const newCollapsed = req.body.collapsed;        Lists.direct.update(          {            _id: paramListId,            boardId: paramBoardId,            archived: false,          },          {            $set: {              collapsed: newCollapsed,            },          },        );        updated = true;      }      // Update wipLimit if provided      if (req.body.wipLimit) {        const newWipLimit = req.body.wipLimit;        Lists.direct.update(          {            _id: paramListId,            boardId: paramBoardId,            archived: false,          },          {            $set: {              wipLimit: newWipLimit,            },          },        );        updated = true;      }      // Check if update is true or false      if (!updated) {        JsonRoutes.sendResult(res, {          code: 404,          data: {            message: 'Error',          },        });        return;      }      JsonRoutes.sendResult(res, {        code: 200,        data: {          _id: paramListId,        },      });    } 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;      const paramListId = req.params.listId;      Authentication.checkBoardAccess(req.userId, paramBoardId);      Lists.remove({ _id: paramListId, boardId: paramBoardId });      JsonRoutes.sendResult(res, {        code: 200,        data: {          _id: paramListId,        },      });    } catch (error) {      JsonRoutes.sendResult(res, {        code: 200,        data: error,      });    }  });}// Position history tracking methodsLists.helpers({  /**   * Track the original position of this list   */  trackOriginalPosition() {    const existingHistory = PositionHistory.findOne({      boardId: this.boardId,      entityType: 'list',      entityId: this._id,    });    if (!existingHistory) {      PositionHistory.insert({        boardId: this.boardId,        entityType: 'list',        entityId: this._id,        originalPosition: {          sort: this.sort,          title: this.title,        },        originalSwimlaneId: this.swimlaneId || null,        originalTitle: this.title,        createdAt: new Date(),        updatedAt: new Date(),      });    }  },  /**   * Get the original position history for this list   */  getOriginalPosition() {    return PositionHistory.findOne({      boardId: this.boardId,      entityType: 'list',      entityId: this._id,    });  },  /**   * Check if this list has moved from its original position   */  hasMovedFromOriginalPosition() {    const history = this.getOriginalPosition();    if (!history) return false;        const currentSwimlaneId = this.swimlaneId || null;    return history.originalPosition.sort !== this.sort ||           history.originalSwimlaneId !== currentSwimlaneId;  },  /**   * Get a description of the original position   */  getOriginalPositionDescription() {    const history = this.getOriginalPosition();    if (!history) return 'No original position data';        const swimlaneInfo = history.originalSwimlaneId ?       ` in swimlane ${history.originalSwimlaneId}` :       ' in default swimlane';    return `Original position: ${history.originalPosition.sort || 0}${swimlaneInfo}`;  },  /**   * Get the effective swimlane ID (for backward compatibility)   */  getEffectiveSwimlaneId() {    return this.swimlaneId || null;  },});export default Lists;
 |