//-- copyright // OpenProject is a project management system. // Copyright (C) 2012-2015 the OpenProject Foundation (OPF) // // This program is free software; you can redistribute it and/or // modify it under the terms of the GNU General Public License version 3. // // OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows: // Copyright (C) 2006-2013 Jean-Philippe Lang // Copyright (C) 2010-2013 the ChiliProject Team // // This program is free software; you can redistribute it and/or // modify it under the terms of the GNU General Public License // as published by the Free Software Foundation; either version 2 // of the License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software // Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. // // See doc/COPYRIGHT.rdoc for more details. //++ import {HalResource} from './hal-resource.service'; import {opApiModule} from '../../../../angular-modules'; import {WorkPackageCacheService} from '../../../work-packages/work-package-cache.service'; import {SchemaCacheService} from './../../../schemas/schema-cache.service'; import {ApiWorkPackagesService} from '../../api-work-packages/api-work-packages.service'; import {CollectionResource, CollectionResourceInterface} from './collection-resource.service'; import {AttachmentCollectionResourceInterface} from './attachment-collection-resource.service'; import {UploadFile} from '../../op-file-upload/op-file-upload.service'; import IQService = angular.IQService; import IPromise = angular.IPromise; import ITimeoutService = angular.ITimeoutService; import {States} from '../../../states.service'; import {SchemaResource} from './schema-resource.service'; import {TypeResource} from './type-resource.service'; import {RelationResourceInterface} from './relation-resource.service'; interface WorkPackageResourceEmbedded { activities: CollectionResourceInterface; ancestors: WorkPackageResourceInterface[]; assignee: HalResource|any; attachments: AttachmentCollectionResourceInterface; author: HalResource|any; availableWatchers: HalResource|any; category: HalResource|any; children: WorkPackageResourceInterface[]; parent: HalResource|any; priority: HalResource|any; project: HalResource|any; relations: CollectionResourceInterface; responsible: HalResource|any; revisions: CollectionResourceInterface|any; status: HalResource|any; timeEntries: HalResource[]|any[]; type: TypeResource; version: HalResource|any; watchers: CollectionResourceInterface; // For regular work packages startDate: string; dueDate: string; // Only for milestones date: string; relatedBy: RelationResourceInterface|null; } interface WorkPackageResourceLinks extends WorkPackageResourceEmbedded { addAttachment(attachment: HalResource): ng.IPromise; addChild(child: HalResource): ng.IPromise; addComment(comment: HalResource): ng.IPromise; addRelation(relation: any): ng.IPromise; addWatcher(watcher: HalResource): ng.IPromise; changeParent(params: any): ng.IPromise; copy(): ng.IPromise; delete(): ng.IPromise; logTime(): ng.IPromise; move(): ng.IPromise; removeWatcher(): ng.IPromise; self(): ng.IPromise; update(payload: any): ng.IPromise; updateImmediately(payload: any): ng.IPromise; watch(): ng.IPromise; } interface WorkPackageLinksObject extends WorkPackageResourceLinks { schema: HalResource; } var $q: IQService; var $stateParams: any; var $timeout: ITimeoutService; var I18n: op.I18n; var states: States; var apiWorkPackages: ApiWorkPackagesService; var wpCacheService: WorkPackageCacheService; var schemaCacheService: SchemaCacheService; var NotificationsService: any; var wpNotificationsService: any; var AttachmentCollectionResource:any; export class WorkPackageResource extends HalResource { // Add index signature for getter this[attr] [attribute:string]:any; public static fromCreateForm(form:any) { var wp = new WorkPackageResource(form.payload.$plain(), true); wp.initializeNewResource(form); return wp; } /** * Create a copy resource from other and the new work package form * @param otherForm The work package form of another work package * @param form Work Package create form */ public static copyFrom(otherForm:any, form:any) { var wp = new WorkPackageResource(otherForm.payload.$plain(), true); // Override values from form payload wp.lockVersion = form.payload.lockVersion; wp.initializeNewResource(form); return wp; } public $embedded: WorkPackageResourceEmbedded; public $links: WorkPackageLinksObject; public $pristine: { [attribute: string]: any } = {}; public parentId: number; public subject: string; public updatedAt: Date; public lockVersion: number; public description: any; public inFlight: boolean; public activities: CollectionResourceInterface; public attachments: AttachmentCollectionResourceInterface; public pendingAttachments: UploadFile[] = []; private form:any; // Keep a reference to an embedded form schema, // if this work package currently has one. private overriddenSchema:SchemaResource|null; public get id():string { return this.$source.id || this.idFromLink; } public static idFromLink(href: string): string { return href.split('/').pop()!; } public get idFromLink(): string { if (this.href) { return WorkPackageResource.idFromLink(this.href); } return ''; } /** * Return ": " if the type is known. */ public get subjectWithType():string { if (this.type) { return `${this.type.name}: ${this.subject}`; } else { return this.subject; } } public get isNew(): boolean { return this.id === 'new'; } public get isMilestone(): boolean { return this.schema.hasOwnProperty('date'); } /** * Returns true if any field is in edition in this resource. */ public get dirty(): boolean { return this.modifiedFields.length > 0; } /** * Returns all modified fields by comparing open $pristine fields. */ public get modifiedFields(): string[] { var modified:string[] = []; angular.forEach(this.$pristine, (value, key) => { var args = [this[key], value]; if ((this as any)[key] instanceof HalResource) { args = args.map(arg => (arg ? arg.$source : arg)); } if (!_.isEqual(args[0], args[1])) { modified.push(key); } }); return modified; } public get isLeaf(): boolean { var children = this.$links.children; return !(children && children.length > 0); } public get isEditable(): boolean { return !!this.$links.update || this.isNew; } /** * Return whether the user is able to upload an attachment. * * If either the `addAttachment` link is provided or the resource is being created, * adding attachments is allowed. */ public get canAddAttachments(): boolean { return !!this.$links.addAttachment || this.isNew; } /** * Initialise the work package resource. * * Make the attachments an `AttachmentCollectionResource`. This should actually * be done automatically, but the backend does not provide typed collections yet. */ protected $initialize(source:any) { super.$initialize(source); var attachments:{$source:any, $loaded:boolean} = this.attachments || {$source: void 0, $loaded: false}; this.attachments = new AttachmentCollectionResource( attachments.$source, attachments.$loaded ); } /** * Remove the given attachment either from the pending attachments or from * the attachment collection, if it is a resource. * * Removing it from the elements array assures that the view gets updated immediately. * If an error occurs, the user gets notified and the attachment is pushed to the elements. */ public removeAttachment(attachment:any) { if (attachment.$isHal) { attachment.delete() .then(() => { this.updateAttachments(); }) .catch((error:any) => { wpNotificationsService.handleErrorResponse(error, this); this.attachments.elements.push(attachment); }); } _.pull(this.attachments.elements, attachment); _.pull(this.pendingAttachments, attachment); } /** * Upload the pending attachments if the work package exists. * Do nothing, if the work package is being created. */ public uploadPendingAttachments():ng.IPromise|void { if (!this.pendingAttachments.length) { return; } const attachments = this.pendingAttachments; this.pendingAttachments = []; return this.uploadAttachments(attachments); } /** * Upload the given attachments, update the resource and notify the user. * Return an updated AttachmentCollectionResource. */ public uploadAttachments(files: UploadFile[]): IPromise { const {uploads, finished} = this.attachments.upload(files); const message = I18n.t('js.label_upload_notification', this); const notification = NotificationsService.addWorkPackageUpload(message, uploads); return finished .then(() => { $timeout(() => NotificationsService.remove(notification), 700); return this.updateAttachments(); }) .catch(error => { wpNotificationsService.handleRawError(error, this); }); } public allowedValuesFor(field:string):ng.IPromise { var deferred = $q.defer(); this.getForm().then((form:any) => { const fieldSchema = form.$embedded.schema[field]; if (!fieldSchema) { deferred.resolve([]); } else if (fieldSchema.allowedValues && fieldSchema.allowedValues['$load']) { let allowedValues = fieldSchema.allowedValues; return allowedValues.$load().then((loadedValues:CollectionResource) => { deferred.resolve(loadedValues.elements); }); } else { deferred.resolve(fieldSchema.allowedValues); } }); return deferred.promise; } public setAllowedValueFor(field:string, value:string|HalResource) { return this.allowedValuesFor(field).then(allowedValues => { let newValue; if ((value as HalResource)['$href']) { newValue = _.find(allowedValues, (entry:any) => entry.$href === (value as HalResource).$href); } else if (allowedValues) { newValue = _.find(allowedValues, (entry:any) => entry === value); } else { newValue = value; } if (newValue) { (this as any)[field] = newValue; } wpCacheService.updateWorkPackage(this); }); } public getForm() { if (!this.form) { this.updateForm(this.$source).catch(error => { NotificationsService.addError(error.message); }); } return this.form; } public updateForm(payload:{[attribute:string]: any}) { // Always resolve form to the latest form // This way, we won't have to actively reset it. // But store the existing form in case of an error. // Because if we get an error, the object returned is not a form // and thus lacks the links the implementation depends upon. var oldForm = this.form; this.form = this.$links.update(payload); var deferred = $q.defer(); this.form .then((form:any) => { // Override the current schema with // the changes from API this.overriddenSchema = form.$embedded.schema; // Take over new values from the form // this resource doesn't know yet. _.defaultsDeep(this.$source, form.$source._embedded.payload); this.$initialize(this.$source); deferred.resolve(form); }) .catch((error:any) => { this.form = oldForm; deferred.reject(error); }); return deferred.promise; } public loadFormSchema() { return this.getForm().then((form:any) => { this.overriddenSchema = form.$embedded.schema; angular.forEach(this.schema, (field, name) => { // Assign only links from schema when an href is set // and field is writable. // (exclude plain properties and null values) const isHalField = field.writable && this[name] && this[name].href; // Assign only values from embedded schema that have an expanded // allowedValues set (type, status, custom field lists, etc.) const hasAllowedValues = Array.isArray(field.allowedValues) && field.allowedValues.length > 0; if (isHalField && hasAllowedValues) { this[name] = _.find(field.allowedValues, {href: this[name].href}) || this[name]; } }); return this.schema; }); } public save() { var deferred = $q.defer(); this.inFlight = true; const wasNew = this.isNew; this.updateForm(this.$source) .then(form => { const payload = this.mergeWithForm(form); const sentValues = Object.keys(this.$pristine); this.$links.updateImmediately(payload) .then((workPackage:WorkPackageResource) => { // Remove the current form and schema, otherwise old form data // might still be used for the next edit field to be edited this.form = null; this.overriddenSchema = null; // Initialize any potentially new HAL values this.$initialize(workPackage); // Ensure the schema is loaded before updating schemaCacheService.ensureLoaded(this).then(() => { this.updateActivities(); if (wasNew) { this.uploadPendingAttachments(); wpCacheService.newWorkPackageCreated(this); } // Remove only those pristine values that were submitted angular.forEach(sentValues, (key) => { delete this.$pristine[key]; }); deferred.resolve(this); }); }) .catch(error => { deferred.reject(error); wpCacheService.updateWorkPackage(this); }) .finally(() => { this.inFlight = false; }); }) .catch(() => { this.inFlight = false; deferred.reject(); }); return deferred.promise; } public storePristine(attribute: string) { if (this.$pristine.hasOwnProperty(attribute)) { return; } this.$pristine[attribute] = angular.copy(this[attribute]); } public restoreFromPristine(attribute: string) { if (this.$pristine.hasOwnProperty(attribute)) { this[attribute] = this.$pristine[attribute]; delete this.$pristine[attribute]; } } public isParentOf(otherWorkPackage:WorkPackageResourceInterface) { return otherWorkPackage.parent.$links.self.$link.href === this.$links.self.$link.href; } private mergeWithForm(form:any) { var plainPayload = form.payload.$plain(); var schema = form.$embedded.schema; // Merge embedded properties from form payload // Do not use properties on this, since they may be incomplete // e.g., when switching to a type that requires a custom field. Object.keys(plainPayload).forEach(key => { if (typeof(schema[key]) === 'object' && schema[key].writable === true) { plainPayload[key] = this[key]; } }); // Merged linked properties from form payload Object.keys(plainPayload._links).forEach(key => { if (typeof(schema[key]) === 'object' && schema[key].writable === true) { var isArray = (schema[key].type || '').startsWith('[]'); if (isArray) { var links:{href:string}[] = []; var val = this[key]; if (val) { var elements = (val.forEach && val) || val.elements; elements.forEach((link:{href:string}) => { if (link.href) { links.push({ href: link.href }); } }); } plainPayload._links[key] = links; } else { var value = this[key] ? this[key].href : null; plainPayload._links[key] = {href: value}; } } }); return plainPayload; } /** * Invalidate a set of linked resources of this work package. * And inform the cache service about the work package update. * * Return a promise that returns the linked resources as properties. * Return a rejected promise, if the resource is not a property of the work package. */ public updateLinkedResources(...resourceNames:string[]): ng.IPromise { const resources: {[id: string]: IPromise} = {}; resourceNames.forEach(name => { const linked = this[name]; resources[name] = linked ? linked.$update() : $q.reject(); }); const promise = $q.all(resources) promise.then(() => { wpCacheService.updateWorkPackage(this); }); return promise; } /** * Get updated activities from the server and inform the cache service about the work * package update. * * Return a promise that returns the activities. Reject, if the work package has * no activities. */ public updateActivities(): IPromise { return this .updateLinkedResources('activities') .then((resources: any) => resources.activities); } /** * Get updated attachments and activities from the server and inform the cache service * about the update. * * Return a promise that returns the attachments. Reject, if the work package has * no attachments. */ public updateAttachments(): IPromise { return this .updateLinkedResources('activities', 'attachments') .then((resources: any) => resources.attachments); } /** * Assign values from the form for a newly created work package resource. * @param form */ public initializeNewResource(form:any) { this.overriddenSchema = form.schema; this.form = $q.when(form); this.$source.id = 'new'; // Set update link to form this['update'] = this.$links.update = form.$links.self; // Use POST /work_packages for saving link this['updateImmediately'] = this.$links.updateImmediately = (payload) => { return apiWorkPackages.createWorkPackage(payload); }; this.parentId = this.parentId || $stateParams.parent_id; } /** * Exclude the schema _link from the linkable Resources. */ public $linkableKeys():string[] { return _.without(super.$linkableKeys(), 'schema'); } /** * Get the current schema, assuming it is either: * 1. Overridden by the current loaded form * 2. Available as a schema state * * If it is neither, an exception is raised. */ public get schema():SchemaResource { if (this.hasOverriddenSchema) { return this.overriddenSchema!; } const state = schemaCacheService.state(this); if (!state.hasValue()) { throw `Accessing schema of ${this.id} without it being loaded.`; } return state.value!; } public get hasOverriddenSchema():boolean { return this.overriddenSchema != null; } } export interface WorkPackageResourceInterface extends WorkPackageResourceLinks, WorkPackageResourceEmbedded, WorkPackageResource { } function wpResource(...args:any[]) { [ $q, $stateParams, $timeout, I18n, states, apiWorkPackages, wpCacheService, schemaCacheService, NotificationsService, wpNotificationsService, AttachmentCollectionResource] = args; return WorkPackageResource; } wpResource.$inject = [ '$q', '$stateParams', '$timeout', 'I18n', 'states', 'apiWorkPackages', 'wpCacheService', 'schemaCacheService', 'NotificationsService', 'wpNotificationsService', 'AttachmentCollectionResource' ]; opApiModule.factory('WorkPackageResource', wpResource);