kanbanworkflowstimelinescrumrubyroadmapproject-planningproject-managementopenprojectangularissue-trackerifcgantt-chartganttbug-trackerboardsbcf
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.
405 lines
12 KiB
405 lines
12 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.
|
|
//++
|
|
|
|
angular
|
|
.module('openproject.services')
|
|
.service('WorkPackageFieldService', WorkPackageFieldService);
|
|
|
|
function WorkPackageFieldService($q, $http, $filter, I18n, WorkPackagesHelper, HookService,
|
|
inplaceEditErrors) {
|
|
|
|
function getSchema(workPackage) {
|
|
if (workPackage.form) {
|
|
return workPackage.form.embedded.schema;
|
|
} else {
|
|
return workPackage.schema;
|
|
}
|
|
}
|
|
|
|
function isEditable(workPackage, field) {
|
|
// no form - no editing
|
|
if (!workPackage.form) {
|
|
return false;
|
|
}
|
|
var schema = getSchema(workPackage);
|
|
// TODO: extract to strategy if new cases arise
|
|
if (field === 'date') {
|
|
// nope
|
|
return schema.props['startDate'].writable && schema.props['dueDate'].writable;
|
|
//return workPackage.schema.props.startDate.writable
|
|
// && workPackage.schema.props.dueDate.writable;
|
|
}
|
|
if(schema.props[field].type === 'Date') {
|
|
return true;
|
|
}
|
|
var isWritable = schema.props[field].writable;
|
|
|
|
// not writable if no embedded allowed values
|
|
if (isWritable && schema.props[field]._links && allowedValuesEmbedded(workPackage, field)) {
|
|
isWritable = getEmbeddedAllowedValues(workPackage, field).length > 0;
|
|
}
|
|
|
|
return isWritable;
|
|
}
|
|
|
|
function isSpecified(workPackage, field) {
|
|
var schema = getSchema(workPackage);
|
|
if (field === 'date') {
|
|
// kind of specified
|
|
return true;
|
|
}
|
|
return !_.isUndefined(schema.props[field]);
|
|
}
|
|
|
|
// under special conditions fields will be shown
|
|
// irregardless if they are empty or not
|
|
// e.g. when an error should trigger the editing state
|
|
// of an empty field after type change
|
|
function isHideable(workPackage, field) {
|
|
if (inplaceEditErrors.errors && inplaceEditErrors.errors[field]) {
|
|
return false;
|
|
}
|
|
return isEmpty(workPackage, field) && !isRequired(workPackage, field);
|
|
}
|
|
|
|
function isMilestone(workPackage) {
|
|
// TODO: this should be written as "only use the form when editing"
|
|
// otherwise always use the simple way
|
|
// currently we don't know the context in which this method is called
|
|
var formAvailable = !_.isUndefined(workPackage.form);
|
|
if (formAvailable) {
|
|
var embedded = workPackage.form.embedded,
|
|
allowedValues = embedded.schema.props.type._embedded.allowedValues,
|
|
currentType = embedded.payload.links.type.props.href;
|
|
return _.some(allowedValues, function(allowedValue) {
|
|
return allowedValue._links.self.href === currentType &&
|
|
allowedValue.isMilestone;
|
|
});
|
|
} else {
|
|
return workPackage.embedded.type.isMilestone;
|
|
}
|
|
}
|
|
|
|
function getValue(workPackage, field, isReadMode) {
|
|
var embedded = !isReadMode && workPackage.form && workPackage.form.embedded.payload;
|
|
var payload = embedded || workPackage;
|
|
|
|
if (field === 'date') {
|
|
if(isMilestone(workPackage)) {
|
|
return payload.props['dueDate'];
|
|
}
|
|
return {
|
|
startDate: payload.props['startDate'],
|
|
dueDate: payload.props['dueDate']
|
|
};
|
|
}
|
|
if (!_.isUndefined(payload.props[field])) {
|
|
return payload.props[field];
|
|
}
|
|
if (WorkPackageFieldService.isEmbedded(payload, field)) {
|
|
return payload.embedded[field];
|
|
}
|
|
|
|
if (payload.links[field] && payload.links[field].props.href !== null) {
|
|
return payload.links[field];
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function allowedValuesEmbedded(workPackage, field) {
|
|
var schema = getSchema(workPackage);
|
|
return _.isArray(schema.props[field]._links.allowedValues);
|
|
}
|
|
|
|
function getEmbeddedAllowedValues(workPackage, field) {
|
|
var options = [];
|
|
var schema = getSchema(workPackage);
|
|
return schema.props[field]._embedded.allowedValues;
|
|
}
|
|
|
|
function getLinkedAllowedValues(workPackage, field) {
|
|
var schema = getSchema(workPackage);
|
|
var href = schema.props[field]._links.allowedValues.href;
|
|
return $http.get(href).then(function(r) {
|
|
var options = [];
|
|
options = _.map(r.data._embedded.elements, function(item) {
|
|
return _.extend({}, item._links.self, { name: item.name });
|
|
});
|
|
return options;
|
|
});
|
|
}
|
|
|
|
function getAllowedValues(workPackage, field) {
|
|
if (allowedValuesEmbedded(workPackage, field)) {
|
|
return $q(function(resolve) {
|
|
resolve(getEmbeddedAllowedValues(workPackage, field));
|
|
});
|
|
} else {
|
|
return getLinkedAllowedValues(workPackage, field);
|
|
}
|
|
}
|
|
|
|
function isRequired(workPackage, field) {
|
|
var schema = getSchema(workPackage);
|
|
if (_.isUndefined(schema.props[field])) {
|
|
return false;
|
|
}
|
|
return schema.props[field].required;
|
|
}
|
|
|
|
function isEmbedded(workPackage, field) {
|
|
return !_.isUndefined(workPackage.embedded[field]);
|
|
}
|
|
|
|
function isSavedAsLink(workPackage, field) {
|
|
return _.isUndefined(workPackage.form.embedded.payload.props[field]);
|
|
}
|
|
|
|
function getLabel(workPackage, field) {
|
|
var schema = getSchema(workPackage);
|
|
if (field === 'date') {
|
|
// special case
|
|
return I18n.t('js.work_packages.properties.date');
|
|
}
|
|
return schema.props[field].name;
|
|
}
|
|
|
|
function isEmpty(workPackage, field) {
|
|
if (field === 'date') {
|
|
return (
|
|
getValue(workPackage, 'startDate') === null &&
|
|
getValue(workPackage, 'dueDate') === null
|
|
);
|
|
}
|
|
var value = WorkPackageFieldService.format(workPackage, field);
|
|
if (value === null || value === '') {
|
|
return true;
|
|
}
|
|
|
|
if (value.html === '') {
|
|
return true;
|
|
}
|
|
|
|
if (field === 'spentTime' && workPackage.props[field] === 'PT0S') {
|
|
return true;
|
|
}
|
|
|
|
if (value.embedded && _.isArray(value.embedded.elements)) {
|
|
return value.embedded.elements.length === 0;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function getInplaceEditStrategy(workPackage, field) {
|
|
var schema = getSchema(workPackage);
|
|
var fieldType = null,
|
|
inplaceType = 'text';
|
|
|
|
if (field === 'date') {
|
|
if(isMilestone(workPackage)) {
|
|
fieldType = 'Date';
|
|
} else {
|
|
fieldType = 'DateRange';
|
|
}
|
|
} else {
|
|
fieldType = schema.props[field].type;
|
|
}
|
|
switch(fieldType) {
|
|
case 'DateRange':
|
|
inplaceType = 'date-range';
|
|
break;
|
|
case 'Date':
|
|
inplaceType = 'date';
|
|
break;
|
|
case 'Float':
|
|
inplaceType = 'float';
|
|
break;
|
|
case 'Integer':
|
|
inplaceType = 'integer';
|
|
break;
|
|
case 'Boolean':
|
|
inplaceType = 'boolean';
|
|
break;
|
|
case 'Formattable':
|
|
if (workPackage.form.embedded.payload.props[field].format === 'textile') {
|
|
inplaceType = 'wiki-textarea';
|
|
} else {
|
|
inplaceType = 'textarea';
|
|
}
|
|
break;
|
|
case 'Duration':
|
|
inplaceType = 'duration';
|
|
break;
|
|
case 'Type':
|
|
inplaceType = 'type';
|
|
break;
|
|
case 'StringObject':
|
|
case 'User':
|
|
case 'Status':
|
|
case 'Priority':
|
|
case 'Category':
|
|
case 'Version':
|
|
inplaceType = 'drop-down';
|
|
break;
|
|
}
|
|
|
|
var typeFromPluginHook = HookService.call('workPackageAttributeEditableType', {
|
|
type: fieldType
|
|
}).pop();
|
|
|
|
if (typeFromPluginHook) {
|
|
inplaceType = typeFromPluginHook;
|
|
}
|
|
return inplaceType;
|
|
}
|
|
|
|
function getInplaceDisplayStrategy(workPackage, field) {
|
|
var schema = getSchema(workPackage);
|
|
var fieldType = null,
|
|
displayStrategy = 'embedded';
|
|
|
|
if (field === 'date') {
|
|
if(isMilestone(workPackage)) {
|
|
fieldType = 'Date';
|
|
} else {
|
|
fieldType = 'DateRange';
|
|
}
|
|
} else if (field === 'spentTime') {
|
|
fieldType = 'SpentTime';
|
|
} else {
|
|
fieldType = schema.props[field].type;
|
|
}
|
|
switch(fieldType) {
|
|
case 'String':
|
|
case 'Integer':
|
|
case 'Float':
|
|
case 'Duration':
|
|
case 'Boolean':
|
|
displayStrategy = 'text';
|
|
break;
|
|
case 'SpentTime':
|
|
displayStrategy = 'spent-time';
|
|
break;
|
|
case 'Formattable':
|
|
displayStrategy = 'wiki-textarea';
|
|
break;
|
|
case 'Version':
|
|
displayStrategy = 'version';
|
|
break;
|
|
case 'User':
|
|
displayStrategy = 'user';
|
|
break;
|
|
case 'DateRange':
|
|
displayStrategy = 'date-range';
|
|
break;
|
|
case 'Date':
|
|
displayStrategy = 'date';
|
|
break;
|
|
}
|
|
|
|
//workPackageOverviewAttributes
|
|
var pluginDirectiveName = HookService.call('workPackageOverviewAttributes', {
|
|
type: fieldType,
|
|
field: field,
|
|
workPackage: workPackage
|
|
})[0];
|
|
if (pluginDirectiveName) {
|
|
displayStrategy = 'dynamic';
|
|
}
|
|
|
|
return displayStrategy;
|
|
}
|
|
|
|
function format(workPackage, field) {
|
|
var schema = getSchema(workPackage);
|
|
if (field === 'date') {
|
|
if(isMilestone(workPackage)) {
|
|
return workPackage.props['dueDate'];
|
|
}
|
|
return {
|
|
startDate: workPackage.props.startDate,
|
|
dueDate: workPackage.props.dueDate,
|
|
noStartDate: I18n.t('js.label_no_start_date'),
|
|
noEndDate: I18n.t('js.label_no_due_date')
|
|
};
|
|
}
|
|
|
|
var value = workPackage.props[field];
|
|
if (_.isUndefined(value)) {
|
|
return WorkPackageFieldService.getValue(workPackage, field, true);
|
|
}
|
|
|
|
if (value === null) {
|
|
return null;
|
|
}
|
|
|
|
var fieldMapping = {
|
|
dueDate: 'date',
|
|
startDate: 'date',
|
|
createdAt: 'datetime',
|
|
updatedAt: 'datetime'
|
|
}[field] || schema.props[field].type;
|
|
|
|
switch(fieldMapping) {
|
|
case('Duration'):
|
|
var hours = moment.duration(value).asHours();
|
|
var formattedHours = $filter('number')(hours, 2);
|
|
return I18n.t('js.units.hour', { count: formattedHours });
|
|
case('Boolean'):
|
|
return value ? I18n.t('js.general_text_yes') : I18n.t('js.general_text_no');
|
|
case('Date'):
|
|
return value;
|
|
case('Float'):
|
|
return $filter('number')(value);
|
|
default:
|
|
return WorkPackagesHelper.formatValue(value, fieldMapping);
|
|
}
|
|
}
|
|
|
|
var WorkPackageFieldService = {
|
|
getSchema: getSchema,
|
|
isEditable: isEditable,
|
|
isRequired: isRequired,
|
|
isSpecified: isSpecified,
|
|
isEmpty: isEmpty,
|
|
isHideable: isHideable,
|
|
isMilestone: isMilestone,
|
|
isEmbedded: isEmbedded,
|
|
isSavedAsLink: isSavedAsLink,
|
|
getValue: getValue,
|
|
getLabel: getLabel,
|
|
getAllowedValues: getAllowedValues,
|
|
allowedValuesEmbedded: allowedValuesEmbedded,
|
|
format: format,
|
|
getInplaceEditStrategy: getInplaceEditStrategy,
|
|
getInplaceDisplayStrategy: getInplaceDisplayStrategy
|
|
};
|
|
|
|
return WorkPackageFieldService;
|
|
}
|
|
|