//-- 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'; import {WorkPackageCreateService} from '../../../wp-create/wp-create.service'; import {WorkPackageNotificationService} from '../../../wp-edit/wp-notification.service'; import {debugLog} from '../../../../helpers/debug_output'; export 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; } export 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:WorkPackageNotificationService; var wpCreate:WorkPackageCreateService; var AttachmentCollectionResource:any; var v3Path:any; export class WorkPackageResource extends HalResource { // Add index signature for getter this[attr] [attribute:string]:any; public $embedded:WorkPackageResourceEmbedded; public $links:WorkPackageLinksObject; public subject:string; public updatedAt:Date; public lockVersion:number; public description:any; public inFlight:boolean; public activities:CollectionResourceInterface; public attachments:AttachmentCollectionResourceInterface; public pendingAttachments:UploadFile[] = []; public overriddenSchema?:SchemaResource; public get id():string { return this.$source.id || this.idFromLink; } /** * Return the ids of all its ancestors, if any */ public get ancestorIds():string { const ancestors = (this as any).ancestors; return ancestors.map((el:WorkPackageResource) => el.id.toString()); } /** * 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'); } 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. */ public $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 as any); 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 as any); }); } /** * Uploads the attachments and reloads the work package when done * Reloading is skipped if no attachment is added */ public uploadAttachmentsAndReload() { const attachmentUpload = this.uploadPendingAttachments(); if (attachmentUpload) { attachmentUpload.then((attachmentsResource) => { if (attachmentsResource.count > 0) { wpCacheService.loadWorkPackage(this.id, true); } }); } } public getSchemaName(name:string):string { if (this.isMilestone && (name === 'startDate' || name === 'dueDate')) { return 'date'; } else { return name; } } public isParentOf(otherWorkPackage:WorkPackageResourceInterface) { return otherWorkPackage.parent.$links.self.$link.href === this.$links.self.$link.href; } /** * 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 as any); }); 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); }; } /** * 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, wpCreate, schemaCacheService, NotificationsService, wpNotificationsService, AttachmentCollectionResource, v3Path] = args; return WorkPackageResource; } wpResource.$inject = [ '$q', '$stateParams', '$timeout', 'I18n', 'states', 'apiWorkPackages', 'wpCacheService', 'wpCreate', 'schemaCacheService', 'NotificationsService', 'wpNotificationsService', 'AttachmentCollectionResource', 'v3Path' ]; opApiModule.factory('WorkPackageResource', wpResource);