import SimpleSchema from 'simpl-schema'; import { union } from 'lodash'; import { ValidatedMethod } from 'meteor/mdg:validated-method'; import { RateLimiterMixin } from 'ddp-rate-limiter-mixin'; import { updateParent } from '/imports/api/parenting/parenting.js'; import { reorderDocs, safeUpdateDocOrder } from '/imports/api/parenting/order.js'; import { RefSchema } from '/imports/api/parenting/ChildSchema.js'; import { assertDocEditPermission } from '/imports/api/sharing/sharingPermissions.js'; import fetchDocByRef from '/imports/api/parenting/fetchDocByRef.js'; import getCollectionByName from '/imports/api/parenting/getCollectionByName.js'; import { recomputeCreatureById } from '/imports/api/creature/computation/methods/recomputeCreature.js'; import recomputeInactiveProperties from '/imports/api/creature/denormalise/recomputeInactiveProperties.js'; import recomputeInventory from '/imports/api/creature/denormalise/recomputeInventory.js'; const organizeDoc = new ValidatedMethod({ name: 'organize.organizeDoc', validate: new SimpleSchema({ docRef: RefSchema, parentRef: RefSchema, order: { type: Number, // Should end in 0.5 to place it reliably between two existing documents }, skipRecompute: { type: Boolean, optional: true, }, }).validator(), mixins: [RateLimiterMixin], rateLimit: { numRequests: 5, timeInterval: 5000, }, run({docRef, parentRef, order, skipRecompute}) { let doc = fetchDocByRef(docRef); let collection = getCollectionByName(docRef.collection); // The user must be able to edit both the doc and its parent to move it // successfully assertDocEditPermission(doc, this.userId); let parent = fetchDocByRef(parentRef); assertDocEditPermission(parent, this.userId); // Change the doc's parent updateParent({docRef, parentRef}); // Change the doc's order to be a half step ahead of its target location collection.update(doc._id, {$set: {order}}, {selector: {type: 'any'}}); // Reorder both ancestors' documents let oldAncestorId = doc.ancestors[0].id; reorderDocs({collection, ancestorId: oldAncestorId}); let newAncestorId = getRootId(parent); if (newAncestorId !== oldAncestorId){ reorderDocs({collection, ancestorId: newAncestorId}); } // Figure out which creatures need to be recalculated after this move let docCreatures = getCreatureAncestors(doc); let parentCreatures = getCreatureAncestors(parent); if (!skipRecompute){ let creaturesToRecompute = union(docCreatures, parentCreatures); // Recompute the creatures creaturesToRecompute.forEach(id => { // The active status of some properties might change due to a change in // ancestry recomputeInactiveProperties(id); if (doc.type === 'container' || doc.type === 'item'){ recomputeInventory(id); } // Some Dependencies depend on ancestry, so a full recompute is needed recomputeCreatureById(id); }); } }, }); const reorderDoc = new ValidatedMethod({ name: 'organize.reorderDoc', validate: new SimpleSchema({ docRef: RefSchema, order: { type: Number, // Should end in 0.5 to place it reliably between two existing documents }, }).validator(), mixins: [RateLimiterMixin], rateLimit: { numRequests: 5, timeInterval: 5000, }, run({docRef, order}) { let doc = fetchDocByRef(docRef); assertDocEditPermission(doc, this.userId); safeUpdateDocOrder({docRef, order}); // Recompute the affected creatures getCreatureAncestors(doc).forEach(id => { recomputeCreatureById(id); }); }, }); function getRootId(doc){ if (doc.ancestors && doc.ancestors.length && doc.ancestors[0]){ return doc.ancestors[0].id; } else { return doc._id; } } function getCreatureAncestors(doc){ let ids = []; if(doc.type === 'pc' || doc.type === 'npc' || doc.type === 'monster'){ ids.push(doc._id); } if (doc.ancestors){ doc.ancestors.forEach(ancestorRef => { if (ancestorRef.collection === 'creatures'){ ids.push(ancestorRef.id); } }); } return ids; } export { organizeDoc, reorderDoc };