|
|
|
//-- 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.
|
|
|
|
//++
|
|
|
|
|
|
|
|
module.exports = function($scope,
|
|
|
|
$rootScope,
|
|
|
|
$state,
|
|
|
|
$stateParams,
|
|
|
|
$location,
|
|
|
|
latestTab,
|
|
|
|
workPackage,
|
|
|
|
I18n,
|
|
|
|
RELATION_TYPES,
|
|
|
|
RELATION_IDENTIFIERS,
|
|
|
|
$filter,
|
|
|
|
$q,
|
|
|
|
WorkPackagesHelper,
|
|
|
|
PathHelper,
|
|
|
|
UsersHelper,
|
|
|
|
ConfigurationService,
|
|
|
|
WorkPackageService,
|
|
|
|
ActivityService,
|
|
|
|
ProjectService,
|
|
|
|
CommonRelationsHandler,
|
|
|
|
ChildrenRelationsHandler,
|
|
|
|
ParentRelationsHandler,
|
|
|
|
WorkPackagesOverviewService,
|
|
|
|
WorkPackageFieldService,
|
|
|
|
EditableFieldsState,
|
|
|
|
WorkPackagesDisplayHelper,
|
|
|
|
NotificationsService,
|
|
|
|
WorkPackageAuthorization,
|
|
|
|
PERMITTED_MORE_MENU_ACTIONS,
|
|
|
|
HookService,
|
|
|
|
$window,
|
|
|
|
WorkPackageAttachmentsService,
|
|
|
|
AuthorisationService
|
|
|
|
) {
|
|
|
|
|
|
|
|
$scope.can = AuthorisationService.can;
|
|
|
|
$scope.cannot = AuthorisationService.cannot;
|
|
|
|
|
|
|
|
$scope.$on('$stateChangeSuccess', function(event, toState){
|
|
|
|
latestTab.registerState(toState.name);
|
|
|
|
});
|
|
|
|
|
|
|
|
$scope.$on('workPackageRefreshRequired', function(e, callback) {
|
|
|
|
refreshWorkPackage(callback);
|
|
|
|
});
|
|
|
|
|
|
|
|
// initialization
|
|
|
|
setWorkPackageScopeProperties(workPackage);
|
|
|
|
|
|
|
|
// stuff copied from details toolbar directive...
|
|
|
|
function getPermittedActions(authorization, permittedMoreMenuActions) {
|
|
|
|
var permittedActions = authorization.permittedActions(permittedMoreMenuActions);
|
|
|
|
var augmentedActions = { };
|
|
|
|
|
|
|
|
angular.forEach(permittedActions, function(value, key) {
|
|
|
|
var css = ['icon-' + key];
|
|
|
|
|
|
|
|
this[key] = { link: value, css: css };
|
|
|
|
}, augmentedActions);
|
|
|
|
|
|
|
|
return augmentedActions;
|
|
|
|
}
|
|
|
|
function getPermittedPluginActions(authorization) {
|
|
|
|
var pluginActions = HookService
|
|
|
|
.call('workPackageDetailsMoreMenu')
|
|
|
|
.reduce(function(previousValue, currentValue) {
|
|
|
|
return angular.extend(previousValue, currentValue);
|
|
|
|
}, { });
|
|
|
|
|
|
|
|
var permittedPluginActions = authorization.permittedActions(Object.keys(pluginActions));
|
|
|
|
var augmentedPluginActions = { };
|
|
|
|
|
|
|
|
angular.forEach(permittedPluginActions, function(value, key) {
|
|
|
|
var css = [].concat(pluginActions[key]);
|
|
|
|
|
|
|
|
if (css.length === 0) {
|
|
|
|
css = ['icon-' + key];
|
|
|
|
}
|
|
|
|
|
|
|
|
this[key] = { link: value, css: css };
|
|
|
|
}, augmentedPluginActions);
|
|
|
|
|
|
|
|
return augmentedPluginActions;
|
|
|
|
}
|
|
|
|
function deleteSelectedWorkPackage() {
|
|
|
|
var promise = WorkPackageService.performBulkDelete([$scope.workPackage.props.id], true);
|
|
|
|
|
|
|
|
promise.success(function() {
|
|
|
|
$state.go('work-packages.list', {projectPath: $scope.projectIdentifier});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
$scope.triggerMoreMenuAction = function(action, link) {
|
|
|
|
switch (action) {
|
|
|
|
case 'delete':
|
|
|
|
deleteSelectedWorkPackage();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
$window.location.href = link;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
var authorization = new WorkPackageAuthorization($scope.workPackage);
|
|
|
|
$scope.permittedActions = angular.extend(getPermittedActions(authorization, PERMITTED_MORE_MENU_ACTIONS),
|
|
|
|
getPermittedPluginActions(authorization));
|
|
|
|
$scope.actionsAvailable = Object.keys($scope.permittedActions).length > 0;
|
|
|
|
|
|
|
|
// END stuff copied from details toolbar directive...
|
|
|
|
|
|
|
|
$scope.I18n = I18n;
|
|
|
|
$scope.$parent.preselectedWorkPackageId = $scope.workPackage.props.id;
|
|
|
|
$scope.maxDescriptionLength = 800;
|
|
|
|
$scope.projectIdentifier = $scope.workPackage.embedded.project.props.identifier;
|
|
|
|
|
|
|
|
|
|
|
|
$scope.watch = function() {
|
|
|
|
if ($scope.isWatched) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$scope.toggleWatchLink
|
|
|
|
.fetch({ ajax: {
|
|
|
|
method: $scope.toggleWatchLink.props.method,
|
|
|
|
href: $scope.toggleWatchLink.props.href,
|
|
|
|
contentType: 'application/json; charset=utf-8',
|
|
|
|
data: JSON.stringify($scope.toggleWatchLink.props.payload)
|
|
|
|
}})
|
|
|
|
.then(refreshWorkPackage, $scope.outputError);
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.unwatch = function() {
|
|
|
|
if (!$scope.isWatched) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$scope.toggleWatchLink
|
|
|
|
.fetch({ ajax: {
|
|
|
|
method: $scope.toggleWatchLink.props.method,
|
|
|
|
href: $scope.toggleWatchLink.props.href,
|
|
|
|
contentType: 'application/json; charset=utf-8',
|
|
|
|
data: JSON.stringify($scope.toggleWatchLink.props.payload)
|
|
|
|
}})
|
|
|
|
.then(refreshWorkPackage, $scope.outputError);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
function fetchProjectTypes() {
|
|
|
|
ProjectService.getProject($scope.projectIdentifier)
|
|
|
|
.then(function(project) {
|
|
|
|
$scope.availableTypes = project.embedded.types;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
fetchProjectTypes();
|
|
|
|
|
|
|
|
function refreshWorkPackage(callback) {
|
|
|
|
WorkPackageService.getWorkPackage($scope.workPackage.props.id)
|
|
|
|
.then(function(workPackage) {
|
|
|
|
setWorkPackageScopeProperties(workPackage);
|
|
|
|
fetchProjectTypes();
|
|
|
|
$scope.$broadcast('workPackageRefreshed');
|
|
|
|
if (angular.isFunction(callback)) {
|
|
|
|
callback(workPackage);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
$rootScope.refreshWorkPackage = refreshWorkPackage; // expose to child controllers
|
|
|
|
$scope.wpPromise = WorkPackageService.getWorkPackage($scope.workPackage.props.id);
|
|
|
|
|
|
|
|
// Inform parent that work package is loaded so back url can be maintained
|
|
|
|
$scope.$emit('workPackgeLoaded');
|
|
|
|
|
|
|
|
function outputMessage(message, isError) {
|
|
|
|
$scope.$emit('flashMessage', {
|
|
|
|
isError: !!isError,
|
|
|
|
text: message
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function outputError(error) {
|
|
|
|
outputMessage(error.message, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
$scope.outputMessage = outputMessage; // expose to child controllers
|
|
|
|
$scope.outputError = outputError; // expose to child controllers
|
|
|
|
|
|
|
|
function aggregateActivities(workPackage) {
|
|
|
|
// Do not yet add any intermittent result to the scope,
|
|
|
|
// as we will get an inconsistent activity view
|
|
|
|
// As we may not what activities will be added at a given time,
|
|
|
|
// let them be aggregated asynchronously.
|
|
|
|
var aggregated = [],
|
|
|
|
totalActivities = 0;
|
|
|
|
|
|
|
|
var aggregate = function(success, activity) {
|
|
|
|
|
|
|
|
if (success === true) {
|
|
|
|
aggregated = aggregated.concat(activity);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (++totalActivities === 2) {
|
|
|
|
$scope.activities = $filter('orderBy')(aggregated,
|
|
|
|
'props.createdAt',
|
|
|
|
$scope.activitiesSortedInDescendingOrder
|
|
|
|
);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
addDisplayedActivities(workPackage, aggregate);
|
|
|
|
addDisplayedRevisions(workPackage, aggregate);
|
|
|
|
}
|
|
|
|
|
|
|
|
function addDisplayedActivities(workPackage, aggregate) {
|
|
|
|
var activities = workPackage.embedded.activities.embedded.elements;
|
|
|
|
aggregate(true, activities);
|
|
|
|
}
|
|
|
|
|
|
|
|
function addDisplayedRevisions(workPackage, aggregate) {
|
|
|
|
var linkedRevisions = workPackage.links.revisions;
|
|
|
|
|
|
|
|
if (linkedRevisions === undefined) {
|
|
|
|
return aggregate();
|
|
|
|
}
|
|
|
|
|
|
|
|
linkedRevisions
|
|
|
|
.fetch()
|
|
|
|
.then(function(data) {
|
|
|
|
aggregate(true, data.embedded.elements);
|
|
|
|
}, aggregate);
|
|
|
|
}
|
|
|
|
|
|
|
|
function setWorkPackageScopeProperties(workPackage){
|
|
|
|
$scope.workPackage = workPackage;
|
|
|
|
$scope.isWatched = !!workPackage.links.unwatch;
|
|
|
|
$scope.displayWatchButton = !!workPackage.links.unwatch || !!workPackage.links.watch;
|
|
|
|
|
|
|
|
if (workPackage.links.watch === undefined) {
|
|
|
|
$scope.toggleWatchLink = workPackage.links.unwatch;
|
|
|
|
} else {
|
|
|
|
$scope.toggleWatchLink = workPackage.links.watch;
|
|
|
|
}
|
|
|
|
|
|
|
|
// autocomplete path
|
|
|
|
var projectId = workPackage.embedded.project.props.id;
|
|
|
|
$scope.autocompletePath = PathHelper.staticWorkPackagesAutocompletePath(projectId);
|
|
|
|
|
|
|
|
// activities and latest activities
|
|
|
|
$scope.activitiesSortedInDescendingOrder = ConfigurationService.commentsSortedInDescendingOrder();
|
|
|
|
$scope.activities = [];
|
|
|
|
aggregateActivities($scope.workPackage);
|
|
|
|
|
|
|
|
// watchers
|
|
|
|
if(workPackage.links.watchers) {
|
|
|
|
$scope.watchers = workPackage.embedded.watchers.embedded.elements;
|
|
|
|
}
|
|
|
|
|
|
|
|
$scope.showStaticPagePath = PathHelper.staticWorkPackagePath($scope.workPackage.props.id);
|
|
|
|
|
|
|
|
// Type
|
|
|
|
$scope.type = workPackage.embedded.type;
|
|
|
|
|
|
|
|
// Author
|
|
|
|
$scope.author = workPackage.embedded.author;
|
|
|
|
$scope.authorPath = PathHelper.staticUserPath($scope.author.props.id);
|
|
|
|
$scope.authorActive = UsersHelper.isActive($scope.author);
|
|
|
|
|
|
|
|
// Attachments
|
|
|
|
$scope.attachments = workPackage.embedded.attachments.embedded.elements;
|
|
|
|
|
|
|
|
// relations
|
|
|
|
$q.all(WorkPackagesHelper.getParent(workPackage)).then(function(parents) {
|
|
|
|
var relationsHandler = new ParentRelationsHandler(workPackage, parents, 'parent');
|
|
|
|
$scope.wpParent = relationsHandler;
|
|
|
|
});
|
|
|
|
|
|
|
|
$q.all(WorkPackagesHelper.getChildren(workPackage)).then(function(children) {
|
|
|
|
var relationsHandler = new ChildrenRelationsHandler(workPackage, children);
|
|
|
|
$scope.wpChildren = relationsHandler;
|
|
|
|
});
|
|
|
|
|
|
|
|
function relationTypeIterator(key) {
|
|
|
|
$q.all(WorkPackagesHelper.getRelationsOfType(
|
|
|
|
workPackage,
|
|
|
|
RELATION_TYPES[key])
|
|
|
|
).then(function(relations) {
|
|
|
|
var relationsHandler = new CommonRelationsHandler(workPackage,
|
|
|
|
relations,
|
|
|
|
RELATION_IDENTIFIERS[key]);
|
|
|
|
$scope[key] = relationsHandler;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
for (var key in RELATION_TYPES) {
|
|
|
|
if (RELATION_TYPES.hasOwnProperty(key)) {
|
|
|
|
relationTypeIterator(key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$scope.toggleWatch = function() {
|
|
|
|
var fetchOptions = {
|
|
|
|
method: $scope.toggleWatchLink.props.method
|
|
|
|
};
|
|
|
|
|
|
|
|
if($scope.toggleWatchLink.props.payload !== undefined) {
|
|
|
|
fetchOptions.contentType = 'application/json; charset=utf-8';
|
|
|
|
fetchOptions.data = JSON.stringify($scope.toggleWatchLink.props.payload);
|
|
|
|
}
|
|
|
|
|
|
|
|
$scope.toggleWatchLink
|
|
|
|
.fetch({ajax: fetchOptions})
|
|
|
|
.then(refreshWorkPackage, outputError);
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.canViewWorkPackageWatchers = function() {
|
|
|
|
return !!($scope.workPackage && $scope.workPackage.embedded.watchers !== undefined);
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.isInitialActivity = ActivityService.isInitialActivity;
|
|
|
|
|
|
|
|
// toggles
|
|
|
|
|
|
|
|
$scope.toggleStates = {
|
|
|
|
hideFullDescription: true,
|
|
|
|
hideAllAttributes: true
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.showWorkPackageDetails = function() {
|
|
|
|
var queryProps = $state.params['query_props'];
|
|
|
|
|
|
|
|
$state.go('work-packages.list.details.overview', {
|
|
|
|
projectPath: $scope.projectIdentifier,
|
|
|
|
workPackageId: $scope.workPackage.props.id,
|
|
|
|
'query_props': queryProps
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.closeShowView = function() {
|
|
|
|
var queryProps = $state.params['query_props'];
|
|
|
|
|
|
|
|
$state.go('work-packages.list', {
|
|
|
|
projectPath: $scope.projectIdentifier,
|
|
|
|
workPackageId: $scope.workPackage.props.id,
|
|
|
|
'query_props': queryProps
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
function getFocusAnchorLabel(tab, workPackage) {
|
|
|
|
var tabLabel = I18n.t('js.work_packages.tabs.' + tab),
|
|
|
|
params = {
|
|
|
|
tab: tabLabel,
|
|
|
|
type: workPackage.props.type,
|
|
|
|
subject: workPackage.props.subject
|
|
|
|
};
|
|
|
|
|
|
|
|
return I18n.t('js.label_work_package_details_you_are_here', params);
|
|
|
|
}
|
|
|
|
|
|
|
|
$scope.focusAnchorLabel = getFocusAnchorLabel(
|
|
|
|
$state.current.url.replace(/\//, ''),
|
|
|
|
$scope.workPackage
|
|
|
|
);
|
|
|
|
|
|
|
|
$scope.editWorkPackage = function() {
|
|
|
|
// TODO: Copied from work-package-details-toolbar-directive
|
|
|
|
// since reusing the directive would break the existing toolbar
|
|
|
|
// markup.
|
|
|
|
var editWorkPackagePath = PathHelper.staticEditWorkPackagePath($scope.workPackage.props.id);
|
|
|
|
var backUrl = '?back_url=' + encodeURIComponent($location.url());
|
|
|
|
|
|
|
|
// TODO: Temporarily going to the old edit dialog until we get in-place editing done
|
|
|
|
window.location = editWorkPackagePath + backUrl;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Stuff copied from DetailsTabOverviewController
|
|
|
|
var vm = this;
|
|
|
|
|
|
|
|
vm.groupedFields = [];
|
|
|
|
vm.hideEmptyFields = true;
|
|
|
|
vm.workPackage = $scope.workPackage;
|
|
|
|
|
|
|
|
vm.isGroupHideable = WorkPackagesDisplayHelper.isGroupHideable;
|
|
|
|
vm.isFieldHideable = WorkPackagesDisplayHelper.isFieldHideable;
|
|
|
|
vm.getLabel = WorkPackagesDisplayHelper.getLabel;
|
|
|
|
vm.isSpecified = WorkPackagesDisplayHelper.isSpecified;
|
|
|
|
vm.hasNiceStar = WorkPackagesDisplayHelper.hasNiceStar;
|
|
|
|
vm.showToggleButton = WorkPackagesDisplayHelper.showToggleButton;
|
|
|
|
vm.filesExist = false;
|
|
|
|
|
|
|
|
activate();
|
|
|
|
|
|
|
|
WorkPackageAttachmentsService.hasAttachments(vm.workPackage).then(function(bool) {
|
|
|
|
vm.filesExist = bool;
|
|
|
|
});
|
|
|
|
|
|
|
|
function activate() {
|
|
|
|
EditableFieldsState.forcedEditState = false;
|
|
|
|
$scope.$watch('workPackage.schema', function(schema) {
|
|
|
|
if (schema) {
|
|
|
|
WorkPackagesDisplayHelper.setFocus();
|
|
|
|
vm.workPackage = $scope.workPackage;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
vm.groupedFields = WorkPackagesOverviewService.getGroupedWorkPackageOverviewAttributes();
|
|
|
|
|
|
|
|
$scope.$watchCollection('vm.workPackage.form', function(form) {
|
|
|
|
var schema = WorkPackageFieldService.getSchema(vm.workPackage);
|
|
|
|
var otherGroup = _.find(vm.groupedFields, {groupName: 'other'});
|
|
|
|
otherGroup.attributes = [];
|
|
|
|
_.forEach(schema.props, function(prop, propName) {
|
|
|
|
if (propName.match(/^customField/)) {
|
|
|
|
otherGroup.attributes.push(propName);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
otherGroup.attributes.sort(function(a, b) {
|
|
|
|
var getLabel = function(field) {
|
|
|
|
return vm.getLabel(vm.workPackage, field);
|
|
|
|
};
|
|
|
|
var left = getLabel(a).toLowerCase(),
|
|
|
|
right = getLabel(b).toLowerCase();
|
|
|
|
return left.localeCompare(right);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
$scope.$on('workPackageUpdatedInEditor', function() {
|
|
|
|
NotificationsService.addSuccess(I18n.t('js.label_successful_update'));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|