OpenProject is the leading open source project management software.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
openproject/frontend/app/work_packages/controllers/work-package-show-controlle...

449 lines
14 KiB

//-- 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);
});
// Listen to the event globally, as listeners are not necessarily
// in the child scope
$rootScope.$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);
}
});
}
$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() {
EditableFieldsState.editAll.toggleState();
};
// 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'));
});
}
};