//-- 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',
function($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;
}
};
$stateProvider
.state('work-packages', {
url: '',
abstract: true,
templateUrl: '/templates/work_packages.html',
controller: 'WorkPackagesController',
resolve: {
latestTab: function($state) {
var stateName = 'work-packages.list.details.overview'; // the default tab
return {
getStateName: function() {
return stateName;
},
registerState: function() {
stateName = $state.current.name;
}
};
}
}
})
.state('work-packages.new', {
url: '/{projects}/{projectPath}/work_packages/new?type',
templateUrl: '/components/routes/partials/work-packages.new.html',
controllerAs: 'vm',
reloadOnSearch: false
})
.state('work-packages.copy', {
url: '/work_packages/{copiedFromWorkPackageId:[0-9]+}/copy',
templateUrl: '/components/routes/partials/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: function ($state, $stateParams, inplaceEditAll) {
inplaceEditAll.start();
$state.go('work-packages.list.details.overview', $stateParams);
}
})
.state('work-packages.show', {
url: '/work_packages/{workPackageId:[0-9]+}?query_props',
templateUrl: '/components/routes/partials/work-packages.show.html',
controller: 'WorkPackageShowController',
controllerAs: 'vm',
resolve: {
workPackage: function(WorkPackageService, $stateParams) {
return WorkPackageService.getWorkPackage($stateParams.workPackageId);
},
// TODO hack, get rid of latestTab in ShowController
latestTab: function($state) {
var stateName = 'work-package.overview'; // the default tab
return {
getStateName: function() {
return stateName;
},
registerState: function() {
stateName = $state.current.name;
}
};
}
},
// 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: function($state, $timeout){
angular.element('body').addClass('action-show');
$timeout(function () {
if ($state.is('work-packages.show')) {
$state.go('work-packages.show.activity');
}
});
},
onExit: function(){
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', {
url: '/relations',
templateUrl: '/templates/work_packages/tabs/relations.html'
})
.state('work-packages.show.watchers', panels.watchers)
.state('work-packages.list', {
url: '/{projects}/{projectPath}/work_packages?query_id&query_props',
controller: 'WorkPackagesListController',
templateUrl: '/components/routes/partials/work-packages.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: function(){
angular.element('body').addClass('action-index');
},
onExit: function(){
angular.element('body').removeClass('action-index');
}
})
.state('work-packages.list.new', {
url: '/create_new?type',
templateUrl: '/components/routes/partials/work-packages.list.new.html',
reloadOnSearch: false
})
.state('work-packages.list.copy', {
url: '/details/{copiedFromWorkPackageId:[0-9]+}/copy',
templateUrl: '/components/routes/partials/work-packages.list.new.html',
reloadOnSearch: false
})
.state('work-packages.list.details', {
url: '/details/{workPackageId:[0-9]+}',
templateUrl: '/components/routes/partials/work-packages.list.details.html',
controller: 'WorkPackageDetailsController',
reloadOnSearch: false,
resolve: {
workPackage: function(WorkPackageService, $stateParams) {
return WorkPackageService.getWorkPackage($stateParams.workPackageId);
}
}
})
.state('work-packages.list.details.overview', {
url: '/overview',
controller: 'DetailsTabOverviewController',
templateUrl: '/templates/work_packages/tabs/overview.html',
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', {
url: '/relations',
templateUrl: '/templates/work_packages/tabs/relations.html',
})
.state('work-packages.list.details.watchers', panels.watchers);
}])
.run([
'$location',
'$rootElement',
'$browser',
'$rootScope',
'$state',
function($location, $rootElement, $browser, $rootScope, $state) {
// 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]', function(event) {
if (!jQuery('body').has('div[ui-view]').length) { return; }
if (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', function(event, toState, toParams){
if (!toParams.projects && toParams.projectPath) {
toParams.projects = 'projects';
$state.go(toState, toParams);
}
});
}
]);