// -- 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') .config(($stateProvider, $urlRouterProvider, $urlMatcherFactoryProvider) => { $urlRouterProvider.when('/work_packages/', '/work_packages'); $urlMatcherFactoryProvider.strictMode(false); var panels = { get watchers() { return { url: '/watchers', template: '' } }, get activity() { return { url: '/activity', template: '' } }, get activityDetails() { var activity = this.activity; activity.url = '#{activity_no:\d+}'; return activity; }, get relations() { return { url: '/relations', templateUrl: '/templates/work_packages/tabs/relations.html' }; } }; $stateProvider .state('work-packages', { url: '', abstract: true, templateUrl: '/components/routing/main/work-packages.html', controller: 'WorkPackagesController' }) .state('work-packages.new', { url: '/{projects}/{projectPath}/work_packages/new?type&parent_id', templateUrl: '/components/routing/main/work-packages.new.html', reloadOnSearch: false }) .state('work-packages.copy', { url: '/work_packages/{copiedFromWorkPackageId:[0-9]+}/copy', templateUrl: '/components/routing/main/work-packages.new.html' }) .state('work-packages.edit', { url: '/{projects}/{projectPath}/work_packages/{workPackageId}/edit', params: { projectPath: {value: null, squash: true}, projects: {value: null, squash: true} }, onEnter: ($state, $stateParams, inplaceEditAll) => { inplaceEditAll.start(); $state.go('work-packages.list.details.overview', $stateParams); } }) .state('work-packages.show', { url: '/work_packages/{workPackageId:[0-9]+}?query_id&query_props', templateUrl: '/components/routing/wp-show/wp.show.html', controller: 'WorkPackageShowController', controllerAs: 'vm', resolve: { workPackage: (WorkPackageService, $stateParams) => { return WorkPackageService.getWorkPackage($stateParams.workPackageId); } }, // HACK // This is to avoid problems with the css depending on which page the // browser starts from (deep-link). As we have CSS rules that change the // layout drastically when on the show action (e.g. position: relative) // and this should not be applied to the other states, we need to remove // the trigger used in the CSS. The correct fix would be to alter the // CSS. onEnter: ($state, $timeout) => { angular.element('body').addClass('action-show'); $timeout(() => { if ($state.is('work-packages.show')) { $state.go('work-packages.show.activity'); } }); }, onExit: () => { angular.element('body').removeClass('action-show'); } }) .state('work-packages.show.activity', panels.activity) .state('work-packages.show.activity.details', panels.activityDetails) .state('work-packages.show.relations', panels.relations) .state('work-packages.show.watchers', panels.watchers) .state('work-packages.list', { url: '/{projects}/{projectPath}/work_packages?query_id&query_props', controller: 'WorkPackagesListController', templateUrl: '/components/routing/wp-list/wp.list.html', params: { // value: null makes the parameter optional // squash: true avoids duplicate slashes when the paramter is not provided projectPath: {value: null, squash: true}, projects: {value: null, squash: true} }, reloadOnSearch: false, // HACK // This is to avoid problems with the css depending on which page the // browser starts from (deep-link). As we have CSS rules that change the // layout drastically when on the index action (e.g. position: absolute, // heigt of footer, ...), and this should not be applied to the other // states, we need to remove the trigger used in the CSS The correct fix // would be to alter the CSS. onEnter: () => { angular.element('body').addClass('action-index'); }, onExit: () => { angular.element('body').removeClass('action-index'); } }) .state('work-packages.list.new', { url: '/create_new?type&parent_id', templateUrl: '/components/routing/wp-list/wp.list.new.html', reloadOnSearch: false }) .state('work-packages.list.copy', { url: '/details/{copiedFromWorkPackageId:[0-9]+}/copy', templateUrl: '/components/routing/wp-list/wp.list.new.html', reloadOnSearch: false }) .state('work-packages.list.details', { url: '/details/{workPackageId:[0-9]+}', templateUrl: '/components/routing/wp-details/wp.list.details.html', controller: 'WorkPackageDetailsController', reloadOnSearch: false, resolve: { workPackage: (WorkPackageService, $stateParams) => { return WorkPackageService.getWorkPackage($stateParams.workPackageId); } } }) .state('work-packages.list.details.overview', { url: '/overview', templateUrl: '/templates/work_packages/tabs/overview.html', controller: 'DetailsTabOverviewController', controllerAs: 'vm', }) .state('work-packages.list.details.activity', panels.activity) .state('work-packages.list.details.activity.details', panels.activityDetails) .state('work-packages.list.details.relations', panels.relations) .state('work-packages.list.details.watchers', panels.watchers); }) .run(($location, $rootElement, $browser, $rootScope, $state, $window) => { // Our application is still a hybrid one, meaning most routes are still // handled by Rails. As such, we disable the default link-hijacking that // Angular's HTML5-mode turns on. $rootElement.off('click'); $rootElement.on('click', 'a[data-ui-route]', (event) => { if (!jQuery('body').has('div[ui-view]').length || event.ctrlKey || event.metaKey || event.which === 2) { return; } // NOTE: making use of event delegation, thus jQuery-only. var elm = jQuery(event.target); var absHref = elm.prop('href'); var rewrittenUrl = $location.$$rewrite(absHref); if (absHref && !elm.attr('target') && rewrittenUrl && !event.isDefaultPrevented()) { event.preventDefault(); if (rewrittenUrl !== $browser.url()) { // update location manually $location.$$parse(rewrittenUrl); $rootScope.$apply(); // hack to work around FF6 bug 684208 when scenario runner clicks on links $window.angular['ff-684208-preventDefault'] = true; } } }); $rootScope.$on('$stateChangeStart', (event, toState, toParams) => { if (!toParams.projects && toParams.projectPath) { toParams.projects = 'projects'; $state.go(toState, toParams); } }); } );