// The core module used for the OpenSlides site and the projector angular.module('OpenSlidesApp.core', []) .config(function(DSProvider) { // Reloads everything after 5 minutes. // TODO: * find a way only to reload things that are still needed DSProvider.defaults.maxAge = 5 * 60 * 1000; // 5 minutes DSProvider.defaults.reapAction = 'none'; DSProvider.defaults.afterReap = function(model, items) { if (items.length > 5) { model.findAll({}, {bypassCache: true}); } else { _.forEach(items, function (item) { model.refresh(item[model.idAttribute]); }); } }; }) .run(function(DS, autoupdate) { autoupdate.on_message(function(data) { // TODO: when MODEL.find() is called after this // a new request is fired. This could be a bug in DS // TODO: Do not send the status code to the client, but make the decission // on the server side. It is an implementation detail, that tornado // sends request to wsgi, which should not concern the client. console.log("Received object: " + data.collection + ", " + data.id); if (data.status_code == 200) { DS.inject(data.collection, data.data); } else if (data.status_code == 404) { DS.eject(data.collection, data.id); } // TODO: handle other statuscodes }); }) .run(function($rootScope, Config) { // Puts the config object into each scope. // TODO: maybe rootscope.config has to set before findAll() is finished Config.findAll().then(function() { $rootScope.config = function(key) { try { return Config.get(key).value; } catch(err) { console.log("Unkown config key: " + key); return '' } } }); }) .factory('autoupdate', function() { var url = location.origin + "/sockjs"; var Autoupdate = { socket: null, message_receivers: [], connect: function() { var autoupdate = this; this.socket = new SockJS(url); this.socket.onmessage = function(event) { _.forEach(autoupdate.message_receivers, function(receiver) { receiver(event.data); }); } this.socket.onclose = function() { setTimeout(autoupdate.connect, 5000); } }, on_message: function(receiver) { this.message_receivers.push(receiver); } }; Autoupdate.connect(); return Autoupdate; }) .factory('Customslide', function(DS) { return DS.defineResource({ name: 'core/customslide', endpoint: '/rest/core/customslide/' }); }) .factory('Tag', function(DS) { return DS.defineResource({ name: 'core/tag', endpoint: '/rest/core/tag/' }); }) .factory('Config', function(DS) { return DS.defineResource({ name: 'config/config', idAttribute: 'key', endpoint: '/rest/config/config/' }); }) .factory('Projector', function(DS) { return DS.defineResource({ name: 'core/projector', endpoint: '/rest/core/projector/', }); }) .run(function(Projector, Config, Tag, Customslide){}); // The core module for the OpenSlides site angular.module('OpenSlidesApp.core.site', ['OpenSlidesApp.core']) .config(function($stateProvider, $urlMatcherFactoryProvider) { // Make the trailing slash optional $urlMatcherFactoryProvider.strictMode(false) // Use stateProvider.decorator to give default values to our states $stateProvider.decorator('views', function(state, parent) { var result = {}, views = parent(state); if (state.abstract || state.data && state.data.extern) { return views; } angular.forEach(views, function(config, name) { // Sets default values for templateUrl var patterns = state.name.split('.'), templateUrl, controller, defaultControllers = { create: 'CreateCtrl', update: 'UpdateCtrl', list: 'ListCtrl', detail: 'DetailCtrl', }; // templateUrl if (_.last(patterns).match(/(create|update)/)) { // When state_patterns is in the form "app.module.create" or // "app.module.update", use the form template. templateUrl = 'static/templates/' + patterns[0] + '/' + patterns[1] + '-form.html' } else { // Replaces the first point through a slash (the app name) var appName = state.name.replace('.', '/'); // Replaces any folowing points though a - templateUrl = 'static/templates/' + appName.replace(/\./g, '-') + '.html'; } config.templateUrl = config.templateUrl || templateUrl; // controller if (patterns.length >= 3) { controller = _.capitalize(patterns[1]) + defaultControllers[_.last(patterns)]; config.controller = config.controller || controller; } result[name] = config; }); return result; }) .decorator('url', function(state, parent) { var defaultUrl; if (state.abstract) { defaultUrl = ''; } else { var patterns = state.name.split('.'), defaultUrls = { create: '/new', update: '/edit', list: '', // The id is expected to be an integer, if not, the url has to // be defined manually detail: '/{id:int}', }; defaultUrl = defaultUrls[_.last(patterns)]; } state.url = state.url || defaultUrl; return parent(state) }); }) .config(function($stateProvider, $locationProvider) { // Core urls $stateProvider .state('dashboard', { url: '/', templateUrl: 'static/templates/dashboard.html' }) .state('projector', { url: '/projector', data: {extern: true}, onEnter: function($window) { $window.location.href = this.url; } }) .state('core', { url: '/core', abstract: true, template: "", }) // tag .state('core.tag', { url: '/tag', abstract: true, template: "", }) .state('core.tag.list', { resolve: { tags: function(Tag) { return Tag.findAll(); } } }) .state('core.tag.create', {}) .state('core.tag.detail.update', { views: { '@core.tag': {} } }); $locationProvider.html5Mode(true); }) // config for ng-fab-form .config(function(ngFabFormProvider) { ngFabFormProvider.extendConfig({ setAsteriskForRequiredLabel: true }); }) // Helper to add ui.router states at runtime. // Needed for the django url_patterns. .provider('runtimeStates', function($stateProvider) { this.$get = function($q, $timeout, $state) { return { addState: function(name, state) { $stateProvider.state(name, state); } } } }) // Load the django url patterns .run(function(runtimeStates, $http) { $http.get('/core/url_patterns/').then(function(data) { for (var pattern in data.data) { runtimeStates.addState(pattern, { 'url': data.data[pattern], data: {extern: true}, onEnter: function($window) { $window.location.href = this.url; } }); } }); }) // options for angular-xeditable .run(function(editableOptions) { editableOptions.theme = 'bs3'; }) // Activate an Element from the Rest-API on the projector // At the moment it only activates item on projector 1 .factory('projectorActivate', function($http) { return function(model, id) { return $http.post( '/rest/core/projector/1/prune_elements/', [{name: model.name, id: id}] ); }; }) .controller("LanguageCtrl", function ($scope, gettextCatalog) { // controller to switch app language // TODO: detect browser language for default language gettextCatalog.setCurrentLanguage('en'); //TODO: for debug only! (helps to find untranslated strings by adding "[MISSING]:") gettextCatalog.debug = true; $scope.switchLanguage = function (lang) { gettextCatalog.setCurrentLanguage(lang); if (lang != 'en') { gettextCatalog.loadRemote("static/i18n/" + lang + ".json"); } } }) .controller("LoginFormCtrl", function ($scope, $modal) { $scope.open = function () { var modalInstance = $modal.open({ animation: true, templateUrl: 'LoginForm.html', controller: 'LoginFormModalCtrl', size: 'sm', }); } }) .controller('LoginFormModalCtrl', function ($scope, $modalInstance, $http, operator) { $scope.login = function () { $http.post( '/users/login/', {'username': $scope.username, 'password': $scope.password} ).success(function(data) { if (data.success) { operator.setUser(data.user_id); $scope.loginFailed = false; $modalInstance.close(); } else { $scope.loginFailed = true; } }); }; $scope.guest = function () { $modalInstance.dismiss('cancel'); }; $scope.cancel = function () { $modalInstance.dismiss('cancel'); }; }) .controller('TagListCtrl', function($scope, Tag) { Tag.bindAll({}, $scope, 'tags'); // setup table sorting $scope.sortColumn = 'name'; $scope.filterPresent = ''; $scope.reverse = false; // function to sort by clicked column $scope.toggleSort = function ( column ) { if ( $scope.sortColumn === column ) { $scope.reverse = !$scope.reverse; } $scope.sortColumn = column; }; // save changed tag $scope.save = function (tag) { Tag.save(tag); }; $scope.delete = function (tag) { //TODO: add confirm message Tag.destroy(tag.id).then( function(success) { //TODO: success message } ); }; }) .controller('TagCreateCtrl', function($scope, $state, Tag) { $scope.tag = {}; $scope.save = function (tag) { Tag.create(tag).then( function(success) { $state.go('core.tag.list'); } ); }; }) .controller('TagUpdateCtrl', function($scope, $state, Tag, tag) { $scope.tag = tag; $scope.save = function (tag) { Tag.save(tag).then( function(success) { $state.go('core.tag.list'); } ); }; }) .directive('osFocusMe', function ($timeout) { return { link: function (scope, element, attrs, model) { $timeout(function () { element[0].focus(); }); } }; }); // The core module for the OpenSlides projector angular.module('OpenSlidesApp.core.projector', ['OpenSlidesApp.core']) // Provider to register slides in a .config() statement. .provider('slides', function() { var slidesMap = {}; this.registerSlide = function(name, config) { slidesMap[name] = config; return this; }; this.$get = function($templateRequest, $q) { var self = this; return { getElements: function(projector) { var elements = []; var factory = this; _.forEach(projector.elements, function(element) { if (element.name in slidesMap) { element.template = slidesMap[element.name].template; elements.push(element); } else { console.log("Unknown slide: " + element.name); } }); return elements; } } }; }) .config(function(slidesProvider) { slidesProvider.registerSlide('core/customslide', { template: 'static/templates/core/slide_customslide.html', }); slidesProvider.registerSlide('core/clock', { template: 'static/templates/core/slide_clock.html', }); }) .filter('osServertime',function() { return function(serverOffset) { var date = new Date(); return date.setTime(date.getTime() - serverOffset); }; }) .controller('ProjectorCtrl', function($scope, Projector, slides) { Projector.find(1).then(function() { $scope.$watch(function () { return Projector.lastModified(1); }, function () { $scope.elements = []; _.forEach(slides.getElements(Projector.get(1)), function(element) { $scope.elements.push(element); }); }); }); }) .controller('SlideCustomSlideCtr', function($scope, Customslide) { // Attention! Each object that is used here has to be dealt on server side. // Add it to the coresponding get_requirements method of the ProjectorElement // class. var id = $scope.element.context.id; Customslide.find(id); Customslide.bindOne(id, $scope, 'customslide'); }) .controller('SlideClockCtr', function($scope) { // Attention! Each object that is used here has to be dealt on server side. // Add it to the coresponding get_requirements method of the ProjectorElement // class. $scope.serverOffset = Date.parse(new Date().toUTCString()) - $scope.element.context.server_time; });