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/components/routing/wp-list/wp-list.controller.ts

290 lines
9.8 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.
// ++
import {WorkPackageCacheService} from '../../work-packages/work-package-cache.service';
import {WorkPackageNotificationService} from '../../wp-edit/wp-notification.service';
import {ErrorResource} from '../../api/api-v3/hal-resources/error-resource.service';
function WorkPackagesListController($scope,
$rootScope,
$state,
$location,
wpNotificationsService: WorkPackageNotificationService,
WorkPackagesTableService,
WorkPackageService,
wpListService,
wpCacheService: WorkPackageCacheService,
ProjectService,
QueryService,
PaginationService,
AuthorisationService,
UrlParamsHelper,
OPERATORS_AND_LABELS_BY_FILTER_TYPE,
loadingIndicator,
I18n) {
$scope.projectIdentifier = $state.params.projectPath || null;
$scope.loadingIndicator = loadingIndicator;
$scope.I18n = I18n;
$scope.text = {
'jump_to_pagination': I18n.t('js.work_packages.jump_marks.pagination'),
'text_jump_to_pagination': I18n.t('js.work_packages.jump_marks.label_pagination')
};
// Setup
function initialSetup() {
$scope.operatorsAndLabelsByFilterType = OPERATORS_AND_LABELS_BY_FILTER_TYPE;
$scope.disableFilters = false;
$scope.disableNewWorkPackage = true;
$scope.queryError = false;
loadingIndicator.mainPage = wpListService.fromQueryParams($state.params, $scope.projectIdentifier)
.then((json:api.ex.WorkPackagesMeta) => {
// Update work package states
wpCacheService.updateWorkPackageList(json.work_packages);
setupPage(json);
})
.catch((result:{ error: ErrorResource|any , json: api.ex.WorkPackagesMeta }) => {
wpNotificationsService.handleErrorResponse(result.error);
setupPage(result.json);
$scope.query.hasError = true;
});
}
function setupQuery(json) {
QueryService.loadAvailableGroupedQueries($scope.projectIdentifier);
QueryService.loadAvailableUnusedColumns($scope.projectIdentifier).then(function(data) {
$scope.availableUnusedColumns = data;
});
var metaData = json.meta;
var queryData = metaData.query;
var columnData = metaData.columns;
var cachedQuery = QueryService.getQuery();
var urlQueryId = $state.params.query_id;
if (cachedQuery && urlQueryId && cachedQuery.id === urlQueryId) {
// Augment current unsaved query with url param data
var updateData = angular.extend(queryData, {columns: columnData});
$scope.query = QueryService.updateQuery(updateData);
} else {
// Set up fresh query from retrieved query meta data
$scope.query = QueryService.initQuery(
$state.params.query_id, queryData, columnData, metaData.export_formats);
if (!!$state.params.query_props) {
$scope.query.dirty = true;
}
}
}
function loadProject() {
if ($scope.projectIdentifier) {
ProjectService.getProject($scope.projectIdentifier).then(function (project) {
$scope.project = project;
$scope.projects = [project];
});
}
}
function setupPage(json) {
// Init query
setupQuery(json);
// Load project
loadProject();
$scope.maintainBackUrl();
// setup table
setupWorkPackagesTable(json);
if (json.work_packages.length) {
WorkPackageService.cache().put('preselectedWorkPackageId', json.work_packages[0].id);
}
}
function setupWorkPackagesTable(json) {
var meta = json.meta,
workPackages = json.work_packages,
bulkLinks = json._bulk_links;
// register data
// table data
WorkPackagesTableService.setColumns($scope.query.columns);
WorkPackagesTableService.addColumnMetaData(meta);
WorkPackagesTableService.setGroupBy($scope.query.groupBy);
WorkPackagesTableService.buildRows(workPackages, $scope.query.groupBy, $state.params.workPackageId);
WorkPackagesTableService.setBulkLinks(bulkLinks);
// query data
QueryService.setTotalEntries(json.resource.total);
// pagination data
PaginationService.setPerPageOptions(meta.per_page_options);
PaginationService.setPerPage(meta.per_page);
PaginationService.setPage(meta.page);
// yield updatable data to scope
$scope.columns = $scope.query.columns;
// Merge new row if it exists
var newRows = WorkPackagesTableService.getRows();
var last = <any> _.last($scope.rows);
if (last && last.object.isNew) {
newRows.push(last);
}
$scope.rows = newRows;
$scope.groupableColumns = WorkPackagesTableService.getGroupableColumns();
$scope.totalEntries = QueryService.getTotalEntries();
$scope.resource = json.resource;
$scope.groupHeaders = WorkPackagesTableService.buildGroupHeaders(json.resource);
// Authorisation
AuthorisationService.initModelAuth('work_package', meta._links);
AuthorisationService.initModelAuth('query', meta.query._links);
}
$scope.setAnchorToNextElement = function () {
// Skip to next when visible, otherwise skip to previous
const selectors = '#pagination--next-link, #pagination--prev-link, #pagination-empty-text';
const visibleLink = jQuery(selectors)
.not(':hidden')
.first();
if (visibleLink.length) {
visibleLink.focus();
}
}
$scope.maintainBackUrl = function () {
$scope.backUrl = $location.url();
};
// Updates
$scope.maintainUrlQueryState = function () {
if ($scope.query) {
$location.search('query_props', UrlParamsHelper.encodeQueryJsonParams($scope.query));
}
};
$scope.loadQuery = function (queryId) {
loadingIndicator.mainPage = $state.go('work-packages.list',
{'query_id': queryId,
'query_props': null});
};
function updateResults() {
$scope.$broadcast('openproject.workPackages.updateResults');
loadingIndicator.mainPage = wpListService.fromQueryInstance($scope.query, $scope.projectIdentifier)
.then(function (json:api.ex.WorkPackagesMeta) {
wpCacheService.updateWorkPackageList(json.work_packages);
setupWorkPackagesTable(json);
});
}
// Go
initialSetup();
$scope.$watch(QueryService.getQueryName, function (queryName) {
$scope.selectedTitle = queryName || I18n.t('js.label_work_package_plural');
});
$scope.$watchCollection(function(){
return {
query_id: $state.params.query_id,
query_props: $state.params.query_props
};
}, function(params) {
if ($scope.query &&
(params.query_id !== $scope.query.id ||
UrlParamsHelper.encodeQueryJsonParams($scope.query) !== params.query_props)) {
initialSetup();
}
});
$rootScope.$on('queryStateChange', function () {
$scope.maintainUrlQueryState();
$scope.maintainBackUrl();
});
$rootScope.$on('workPackagesRefreshRequired', function () {
updateResults();
});
$rootScope.$on('workPackagesRefreshInBackground', function () {
wpListService.fromQueryInstance($scope.query, $scope.projectIdentifier)
.then(function (json:api.ex.WorkPackagesMeta) {
var rowLookup = _.indexBy($scope.rows, (row:any) => row.object.id);
// Merge based on id and lockVersion
angular.forEach(json.work_packages, (fresh, i) => {
var staleRow = rowLookup[fresh.id];
if (staleRow && staleRow.object.lockVersion === fresh.lockVersion) {
json.work_packages[i] = staleRow.object;
}
});
$scope.$broadcast('openproject.workPackages.updateResults');
$scope.$evalAsync(_ => setupWorkPackagesTable(json));
});
});
$rootScope.$on('queryClearRequired', _ => wpListService.clearUrlQueryParams);
function nextAvailableWorkPackage() {
var selected = WorkPackageService.cache().get('preselectedWorkPackageId');
return selected || $scope.rows.first().object.id;
}
$scope.nextAvailableWorkPackage = nextAvailableWorkPackage;
$scope.openWorkPackageInFullView = function (id, force) {
if (force || $state.current.url !== '') {
var params = {
workPackageId: id
}
loadingIndicator.mainPage = $state.go(
'work-packages.show',
angular.extend($state.params, params)
);
}
};
}
angular
.module('openproject.workPackages.controllers')
.controller('WorkPackagesListController', WorkPackagesListController);