| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362 | CustomFields = new Mongo.Collection('customFields');/** * A custom field on a card in the board */CustomFields.attachSchema(  new SimpleSchema({    boardIds: {      /**       * the ID of the board       */      type: [String],    },    name: {      /**       * name of the custom field       */      type: String,    },    type: {      /**       * type of the custom field       */      type: String,      allowedValues: ['text', 'number', 'date', 'dropdown', 'currency'],    },    settings: {      /**       * settings of the custom field       */      type: Object,    },    'settings.currencyCode': {      type: String,      optional: true,    },    'settings.dropdownItems': {      /**       * list of drop down items objects       */      type: [Object],      optional: true,    },    'settings.dropdownItems.$': {      type: new SimpleSchema({        _id: {          /**           * ID of the drop down item           */          type: String,        },        name: {          /**           * name of the drop down item           */          type: String,        },      }),    },    showOnCard: {      /**       * should we show on the cards this custom field       */      type: Boolean,    },    automaticallyOnCard: {      /**       * should the custom fields automatically be added on cards?       */      type: Boolean,    },    showLabelOnMiniCard: {      /**       * should the label of the custom field be shown on minicards?       */      type: Boolean,    },    createdAt: {      type: Date,      optional: true,      // eslint-disable-next-line consistent-return      autoValue() {        if (this.isInsert) {          return new Date();        } else if (this.isUpsert) {          return { $setOnInsert: 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();        }      },    },  }),);CustomFields.mutations({  addBoard(boardId) {    if (boardId) {      return {        $push: {          boardIds: boardId,        },      };    } else {      return null;    }  },});CustomFields.allow({  insert(userId, doc) {    return allowIsAnyBoardMember(      userId,      Boards.find({        _id: { $in: doc.boardIds },      }).fetch(),    );  },  update(userId, doc) {    return allowIsAnyBoardMember(      userId,      Boards.find({        _id: { $in: doc.boardIds },      }).fetch(),    );  },  remove(userId, doc) {    return allowIsAnyBoardMember(      userId,      Boards.find({        _id: { $in: doc.boardIds },      }).fetch(),    );  },  fetch: ['userId', 'boardIds'],});// not sure if we need this?//CustomFields.hookOptions.after.update = { fetchPrevious: false };function customFieldCreation(userId, doc) {  Activities.insert({    userId,    activityType: 'createCustomField',    boardId: doc.boardIds[0], // We are creating a customField, it has only one boardId    customFieldId: doc._id,  });}function customFieldDeletion(userId, doc) {  Activities.insert({    userId,    activityType: 'deleteCustomField',    boardId: doc.boardIds[0], // We are creating a customField, it has only one boardId    customFieldId: doc._id,  });}// This has some bug, it does not show edited customField value at Outgoing Webhook,// instead it shows undefined, and no listId and swimlaneId.function customFieldEdit(userId, doc) {  const card = Cards.findOne(doc.cardId);  const customFieldValue = Activities.findOne({ customFieldId: doc._id }).value;  Activities.insert({    userId,    activityType: 'setCustomField',    boardId: doc.boardIds[0], // We are creating a customField, it has only one boardId    customFieldId: doc._id,    customFieldValue,    listId: doc.listId,    swimlaneId: doc.swimlaneId,  });}if (Meteor.isServer) {  Meteor.startup(() => {    CustomFields._collection._ensureIndex({ modifiedAt: -1 });    CustomFields._collection._ensureIndex({ boardIds: 1 });  });  CustomFields.after.insert((userId, doc) => {    customFieldCreation(userId, doc);  });  CustomFields.before.update((userId, doc, fieldNames, modifier) => {    if (_.contains(fieldNames, 'boardIds') && modifier.$pull) {      Cards.update(        { boardId: modifier.$pull.boardIds, 'customFields._id': doc._id },        { $pull: { customFields: { _id: doc._id } } },        { multi: true },      );      customFieldEdit(userId, doc);      Activities.remove({        customFieldId: doc._id,        boardId: modifier.$pull.boardIds,        listId: doc.listId,        swimlaneId: doc.swimlaneId,      });    } else if (_.contains(fieldNames, 'boardIds') && modifier.$push) {      Activities.insert({        userId,        activityType: 'createCustomField',        boardId: modifier.$push.boardIds,        customFieldId: doc._id,      });    }  });  CustomFields.before.remove((userId, doc) => {    customFieldDeletion(userId, doc);    Activities.remove({      customFieldId: doc._id,    });    Cards.update(      { boardId: { $in: doc.boardIds }, 'customFields._id': doc._id },      { $pull: { customFields: { _id: doc._id } } },      { multi: true },    );  });}//CUSTOM FIELD REST APIif (Meteor.isServer) {  /**   * @operation get_all_custom_fields   * @summary Get the list of Custom Fields attached to a board   *   * @param {string} boardID the ID of the board   * @return_type [{_id: string,   *                name: string,   *                type: string}]   */  JsonRoutes.add('GET', '/api/boards/:boardId/custom-fields', function(    req,    res,  ) {    Authentication.checkUserId(req.userId);    const paramBoardId = req.params.boardId;    JsonRoutes.sendResult(res, {      code: 200,      data: CustomFields.find({ boardIds: { $in: [paramBoardId] } }).map(        function(cf) {          return {            _id: cf._id,            name: cf.name,            type: cf.type,          };        },      ),    });  });  /**   * @operation get_custom_field   * @summary Get a Custom Fields attached to a board   *   * @param {string} boardID the ID of the board   * @param {string} customFieldId the ID of the custom field   * @return_type CustomFields   */  JsonRoutes.add(    'GET',    '/api/boards/:boardId/custom-fields/:customFieldId',    function(req, res) {      Authentication.checkUserId(req.userId);      const paramBoardId = req.params.boardId;      const paramCustomFieldId = req.params.customFieldId;      JsonRoutes.sendResult(res, {        code: 200,        data: CustomFields.findOne({          _id: paramCustomFieldId,          boardIds: { $in: [paramBoardId] },        }),      });    },  );  /**   * @operation new_custom_field   * @summary Create a Custom Field   *   * @param {string} boardID the ID of the board   * @param {string} name the name of the custom field   * @param {string} type the type of the custom field   * @param {string} settings the settings object of the custom field   * @param {boolean} showOnCard should we show the custom field on cards?   * @param {boolean} automaticallyOnCard should the custom fields automatically be added on cards?   * @param {boolean} showLabelOnMiniCard should the label of the custom field be shown on minicards?   * @return_type {_id: string}   */  JsonRoutes.add('POST', '/api/boards/:boardId/custom-fields', function(    req,    res,  ) {    Authentication.checkUserId(req.userId);    const paramBoardId = req.params.boardId;    const board = Boards.findOne({ _id: paramBoardId });    const id = CustomFields.direct.insert({      name: req.body.name,      type: req.body.type,      settings: req.body.settings,      showOnCard: req.body.showOnCard,      automaticallyOnCard: req.body.automaticallyOnCard,      showLabelOnMiniCard: req.body.showLabelOnMiniCard,      boardIds: [board._id],    });    const customField = CustomFields.findOne({      _id: id,      boardIds: { $in: [paramBoardId] },    });    customFieldCreation(req.body.authorId, customField);    JsonRoutes.sendResult(res, {      code: 200,      data: {        _id: id,      },    });  });  /**   * @operation delete_custom_field   * @summary Delete a Custom Fields attached to a board   *   * @description The Custom Field can't be retrieved after this operation   *   * @param {string} boardID the ID of the board   * @param {string} customFieldId the ID of the custom field   * @return_type {_id: string}   */  JsonRoutes.add(    'DELETE',    '/api/boards/:boardId/custom-fields/:customFieldId',    function(req, res) {      Authentication.checkUserId(req.userId);      const paramBoardId = req.params.boardId;      const id = req.params.customFieldId;      CustomFields.remove({ _id: id, boardIds: { $in: [paramBoardId] } });      JsonRoutes.sendResult(res, {        code: 200,        data: {          _id: id,        },      });    },  );}export default CustomFields;
 |