controllers.js 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891
  1. 'use strict';
  2. /* Controllers; their method are available where specified with the ng-controller
  3. * directive or for a given route/state (see app.js). They use some services to
  4. * connect to the backend (see services.js). */
  5. var eventManControllers = angular.module('eventManControllers', []);
  6. /* A controller that can be used to navigate. */
  7. eventManControllers.controller('NavigationCtrl', ['$scope', '$rootScope', '$location', 'Setting', '$state',
  8. function ($scope, $rootScope, $location, Setting, $state) {
  9. $scope.logo = {};
  10. $scope.getLocation = function() {
  11. return $location.absUrl();
  12. };
  13. $scope.go = function(url) {
  14. $location.url(url);
  15. };
  16. Setting.query({setting: 'logo'}, function(data) {
  17. if (data && data.length) {
  18. $scope.logo = data[0];
  19. }
  20. });
  21. $scope.isActive = function(view) {
  22. if (view === $location.path()) {
  23. return true;
  24. }
  25. if (view[view.length-1] !== '/') {
  26. view = view + '/';
  27. }
  28. return $location.path().indexOf(view) == 0;
  29. };
  30. }]
  31. );
  32. /* Controller for a group of date and time pickers. */
  33. eventManControllers.controller('DatetimePickerCtrl', ['$scope',
  34. function ($scope) {
  35. $scope.open = function() {
  36. $scope.opened = true;
  37. };
  38. }]
  39. );
  40. /* Controller for modals. */
  41. eventManControllers.controller('ModalConfirmInstanceCtrl', ['$scope', '$uibModalInstance', 'message',
  42. function ($scope, $uibModalInstance, message) {
  43. $scope.message = message;
  44. $scope.ok = function () {
  45. $uibModalInstance.close($scope);
  46. };
  47. $scope.cancel = function () {
  48. $uibModalInstance.dismiss('cancel');
  49. };
  50. }]
  51. );
  52. eventManControllers.controller('EventsListCtrl', ['$scope', 'Event', 'EventTicket', '$uibModal', '$log', '$translate', '$rootScope', '$state', '$filter', 'toaster',
  53. function ($scope, Event, EventTicket, $uibModal, $log, $translate, $rootScope, $state, $filter, toaster) {
  54. $scope.query = '';
  55. $scope.tickets = [];
  56. $scope.eventsOrderProp = "-begin_date";
  57. $scope.ticketsOrderProp = ["name", "surname"];
  58. $scope.groupByEmail = false;
  59. $scope.shownItems = [];
  60. $scope.currentPage = 1;
  61. $scope.itemsPerPage = 10;
  62. $scope.filteredLength = 0;
  63. $scope.maxPaginationSize = 10;
  64. $scope.events = Event.all({_summary: true}, function(events) {
  65. if (events && $state.is('tickets')) {
  66. angular.forEach(events, function(evt, idx) {
  67. var evt_tickets = (evt.tickets || []).slice(0);
  68. angular.forEach(evt_tickets, function(obj, obj_idx) {
  69. obj.event_title = evt.title;
  70. obj.event_id = evt._id;
  71. });
  72. $scope.tickets.push.apply($scope.tickets, evt_tickets || []);
  73. });
  74. $scope.filterTickets();
  75. }
  76. });
  77. $scope.filterTickets = function() {
  78. var tickets = angular.copy($scope.tickets || []);
  79. if ($scope.groupByEmail) {
  80. var newDict = {};
  81. var newList = [];
  82. angular.forEach(tickets, function(item, idx) {
  83. if (!newDict[item.email]) {
  84. newDict[item.email] = {};
  85. newDict[item.email]['name'] = item.name;
  86. newDict[item.email]['surname'] = item.surname;
  87. newDict[item.email]['email'] = item.email;
  88. newDict[item.email]['job title'] = item.job;
  89. newDict[item.email]['company'] = item.company;
  90. newDict[item.email]['tickets'] = [];
  91. }
  92. newDict[item.email]['tickets'].push(item);
  93. });
  94. angular.forEach(newDict, function(value, key) {
  95. newList.push(value);
  96. });
  97. tickets = newList;
  98. }
  99. tickets = $filter('splittedFilter')(tickets, $scope.query);
  100. tickets = $filter('orderBy')(tickets, $scope.ticketsOrderProp);
  101. $scope.filteredLength = tickets.length;
  102. tickets = $filter('pagination')(tickets, $scope.currentPage, $scope.itemsPerPage);
  103. $scope.shownItems = tickets;
  104. };
  105. $scope.$watch('query', function() {
  106. if (!$scope.query) {
  107. $scope.currentPage = 1;
  108. }
  109. $scope.filterTickets();
  110. });
  111. $scope.$watch('groupByEmail', function() {
  112. $scope.filterTickets();
  113. });
  114. $scope.$watch('currentPage + itemsPerPage', function() {
  115. $scope.filterTickets();
  116. });
  117. $scope.confirm_delete = 'Do you really want to delete this event?';
  118. $scope.confirm_delete_all_tickets = 'Do you really want to delete all tickets from this event?';
  119. $scope.deleted_all_tickets = 'successfully removed all tickets from event';
  120. $rootScope.$on('$translateChangeSuccess', function () {
  121. $translate('Do you really want to delete this event?').then(function (translation) {
  122. $scope.confirm_delete = translation;
  123. });
  124. $translate('Do you really want to delete all tickets from this event?').then(function (translation) {
  125. $scope.confirm_delete_all_tickets = translation;
  126. });
  127. $translate('successfully removed all tickets from event').then(function (translation) {
  128. $scope.deleted_all_tickets = translation;
  129. });
  130. });
  131. $scope.deleteEvent = function(_id) {
  132. var modalInstance = $uibModal.open({
  133. scope: $scope,
  134. templateUrl: 'modal-confirm-action.html',
  135. controller: 'ModalConfirmInstanceCtrl',
  136. resolve: {
  137. message: function() { return $scope.confirm_delete; }
  138. }
  139. });
  140. modalInstance.result.then(function() {
  141. Event.delete({'id': _id}, function() {
  142. $scope.events = Event.all();
  143. });
  144. });
  145. };
  146. $scope.deleteAllTickets = function(_id) {
  147. var modalInstance = $uibModal.open({
  148. scope: $scope,
  149. templateUrl: 'modal-confirm-action.html',
  150. controller: 'ModalConfirmInstanceCtrl',
  151. resolve: {
  152. message: function() { return $scope.confirm_delete_all_tickets; }
  153. }
  154. });
  155. modalInstance.result.then(function() {
  156. EventTicket.delete({
  157. event_id: _id
  158. }, function() {
  159. toaster.pop({type: 'error', title: $scope.deleted_all_tickets});
  160. $scope.events = Event.all();
  161. });
  162. });
  163. };
  164. $scope.updateOrded = function(key) {
  165. var new_order = [key];
  166. var inv_key;
  167. if (key && key[0] === '-') {
  168. inv_key = key.substring(1);
  169. } else {
  170. inv_key = '-' + key;
  171. }
  172. angular.forEach($scope.ticketsOrderProp,
  173. function(value, idx) {
  174. if (value !== key && value !== inv_key) {
  175. new_order.push(value);
  176. }
  177. }
  178. );
  179. $scope.ticketsOrderProp = new_order;
  180. $scope.filterTickets();
  181. };
  182. }]
  183. );
  184. eventManControllers.controller('EventDetailsCtrl', ['$scope', '$state', 'Event', '$log', '$translate', '$rootScope',
  185. function ($scope, $state, Event, $log, $translate, $rootScope) {
  186. $scope.event = {};
  187. $scope.event.tickets = [];
  188. $scope.event.formSchema = {};
  189. $scope.eventFormDisabled = false;
  190. if ($state.params.id) {
  191. var params = angular.copy($state.params);
  192. params['_summary'] = true;
  193. $scope.event = Event.get(params);
  194. if ($state.is('event.view') || !$rootScope.hasPermission('event|update')) {
  195. $scope.eventFormDisabled = true;
  196. }
  197. }
  198. // store a new Event or update an existing one
  199. $scope.save = function() {
  200. // avoid override of event.tickets list.
  201. var this_event = angular.copy($scope.event);
  202. if (this_event.tickets) {
  203. delete this_event.tickets;
  204. }
  205. if (this_event._id === undefined) {
  206. $scope.event = Event.save(this_event);
  207. } else {
  208. $scope.event = Event.update(this_event);
  209. }
  210. $scope.eventForm.$setPristine(false);
  211. };
  212. $scope.saveForm = function(easyFormGeneratorModel) {
  213. $scope.event.formSchema = easyFormGeneratorModel;
  214. $scope.save();
  215. };
  216. }]
  217. );
  218. eventManControllers.controller('EventTicketsCtrl', ['$scope', '$state', 'Event', 'EventTicket', 'Setting', '$log', '$translate', '$rootScope', 'EventUpdates', '$uibModal', '$filter', 'toaster',
  219. function ($scope, $state, Event, EventTicket, Setting, $log, $translate, $rootScope, EventUpdates, $uibModal, $filter, toaster) {
  220. $scope.ticketsOrder = ["name", "surname"];
  221. $scope.countAttendees = 0;
  222. $scope.query = '';
  223. $scope.event = {};
  224. $scope.event.tickets = [];
  225. $scope.shownItems = [];
  226. $scope.ticket = {}; // current ticket, for the event.ticket.* states
  227. $scope.tickets = []; // list of all tickets, for the 'tickets' state
  228. $scope.filteredTickets = [];
  229. $scope.formSchema = {};
  230. $scope.formData = {};
  231. $scope.guiOptions = {dangerousActionsEnabled: false};
  232. $scope.customFields = Setting.query({setting: 'ticket_custom_field', in_event_details: true});
  233. $scope.registeredFilterOptions = {all: false};
  234. $scope.formFieldsMap = {};
  235. $scope.formFieldsMapRev = {};
  236. $scope.currentPage = 1;
  237. $scope.itemsPerPage = 10;
  238. $scope.filteredLength = 0;
  239. $scope.maxPaginationSize = 10;
  240. $scope.maxAllPersons = 10;
  241. $scope.filterTickets = function() {
  242. var tickets = $scope.event.tickets || [];
  243. tickets = $filter('splittedFilter')(tickets, $scope.query);
  244. tickets = $filter('registeredFilter')(tickets, $scope.registeredFilterOptions);
  245. tickets = $filter('orderBy')(tickets, $scope.ticketsOrder);
  246. $scope.filteredTickets = angular.copy(tickets);
  247. $scope.filteredLength = $scope.filteredTickets.length;
  248. tickets = $filter('pagination')(tickets, $scope.currentPage, $scope.itemsPerPage);
  249. $scope.shownItems = tickets;
  250. $scope.updateCSV();
  251. };
  252. $scope.$watch('query', function() {
  253. if (!$scope.query) {
  254. $scope.currentPage = 1;
  255. }
  256. $scope.filterTickets();
  257. });
  258. $scope.$watchCollection('registeredFilterOptions', function() {
  259. $scope.filterTickets();
  260. });
  261. $scope.$watch('currentPage + itemsPerPage', function() {
  262. $scope.filterTickets();
  263. });
  264. if ($state.params.id) {
  265. $scope.event = Event.get({id: $state.params.id}, function(data) {
  266. $scope.$watchCollection(function() {
  267. return $scope.event.tickets;
  268. }, function(new_collection, old_collection) {
  269. $scope.calcAttendees();
  270. $scope.filterTickets();
  271. }
  272. );
  273. if (!(data && data.formSchema)) {
  274. return;
  275. }
  276. $scope.formSchema = data.formSchema.edaFieldsModel;
  277. $scope.extractFormFields(data.formSchema.formlyFieldsModel);
  278. // Editing an existing ticket
  279. if ($state.params.ticket_id) {
  280. EventTicket.get({id: $state.params.id, ticket_id: $state.params.ticket_id}, function(data) {
  281. $scope.ticket = data;
  282. angular.forEach(data, function(value, key) {
  283. if (!$scope.formFieldsMapRev[key]) {
  284. return;
  285. }
  286. $scope.formData[$scope.formFieldsMapRev[key]] = value;
  287. });
  288. });
  289. }
  290. });
  291. // Managing the list of tickets.
  292. if ($state.is('event.tickets')) {
  293. $scope.allPersons = Event.group_persons({id: $state.params.id});
  294. // Handle WebSocket connection used to update the list of tickets.
  295. $scope.EventUpdates = EventUpdates;
  296. $scope.EventUpdates.open();
  297. $scope.$watchCollection(function() {
  298. return $scope.EventUpdates.data;
  299. }, function(new_collection, old_collection) {
  300. if (!($scope.EventUpdates.data && $scope.EventUpdates.data.update)) {
  301. $log.debug('no data received from the WebSocket');
  302. return;
  303. }
  304. var data = $scope.EventUpdates.data.update;
  305. $log.debug('received ' + data.action + ' action from websocket source ' + data.uuid + ' . Full data:');
  306. $log.debug(data);
  307. if ($rootScope.app_uuid == data.uuid) {
  308. $log.debug('do not process our own message');
  309. return false;
  310. }
  311. if (data.error && data.message && $scope.info.user.username == data.username) {
  312. if (data.searchFor) {
  313. $scope.query = angular.copy(data.searchFor);
  314. }
  315. toaster.pop({type: 'error', title: 'Error', body: data.message, timeout: 0, showCloseButton: true});
  316. return;
  317. }
  318. if (!$scope.event.tickets) {
  319. $scope.event.tickets = [];
  320. }
  321. var ticket_id = data._id || (data.ticket && data.ticket._id);
  322. var ticket_idx = $scope.event.tickets.findIndex(function(el, idx, array) {
  323. return ticket_id && (ticket_id == el._id);
  324. });
  325. if (ticket_idx != -1) {
  326. $log.debug('_id ' + data._id + ' found');
  327. } else {
  328. $log.debug('_id ' + data._id + ' not found');
  329. }
  330. if (data.action == 'update' && ticket_idx != -1 && $scope.event.tickets[ticket_idx] != data.ticket) {
  331. // if we're updating the 'attended' key and the action came from us (same user, possibly on
  332. // a different station), also show a message.
  333. if (data.ticket.attended != $scope.event.tickets[ticket_idx].attended &&
  334. $scope.info.user.username == data.username) {
  335. $scope.showAttendedMessage(data.ticket, data.ticket.attended);
  336. }
  337. $scope.event.tickets.splice(ticket_idx, 1, data.ticket);
  338. } else if (data.action == 'add' && ticket_idx == -1) {
  339. $scope._localAddTicket(data.ticket);
  340. } else if (data.action == 'delete' && ticket_idx != -1) {
  341. $scope._localRemoveTicket({_id: data._id});
  342. }
  343. }
  344. );
  345. /* event listners; needed because otherwise, adding a ticket with the Quick add form,
  346. * we'd be changing the $scope outside of the AngularJS's $digest. */
  347. $scope.$on('event:ticket:new', function(evt, ticket, callback) {
  348. $scope._localAddTicket(ticket);
  349. if (callback) {
  350. callback(ticket);
  351. }
  352. });
  353. $scope.$on('event:ticket:update', function(evt, ticket) {
  354. if (!$scope.event.tickets) {
  355. $scope.event.tickets = [];
  356. }
  357. var ticket_idx = $scope.event.tickets.findIndex(function(el, idx, array) {
  358. return ticket._id == el._id;
  359. });
  360. if (ticket_idx == -1) {
  361. $log.debug('ticket not present: not updated');
  362. return false;
  363. }
  364. $scope.event.tickets.splice(ticket_idx, 1, ticket);
  365. });
  366. $scope.$on('event:ticket:set-attr', function(evt, ticket, key, value, callback, hideMessage) {
  367. $scope.setTicketAttribute(ticket, key, value, callback, hideMessage);
  368. });
  369. }
  370. } else if ($state.is('tickets')) {
  371. $scope.tickets = EventTicket.all();
  372. }
  373. $scope.calcAttendees = function() {
  374. if (!($scope.event && $scope.event.tickets)) {
  375. $scope.countAttendees = 0;
  376. return;
  377. }
  378. var attendees = 0;
  379. angular.forEach($scope.event.tickets, function(value, key) {
  380. if (value.attended && !value.cancelled) {
  381. attendees += 1;
  382. }
  383. });
  384. $scope.countAttendees = attendees;
  385. };
  386. /* Stuff to do when a ticket is added, modified or removed locally. */
  387. $scope._localAddTicket = function(ticket, original_ticket) {
  388. if (!$state.is('event.tickets')) {
  389. return true;
  390. }
  391. var ret = true;
  392. if (!$scope.event.tickets) {
  393. $scope.event.tickets = [];
  394. }
  395. var ticket_idx = $scope.event.tickets.findIndex(function(el, idx, array) {
  396. return ticket._id == el._id;
  397. });
  398. if (ticket_idx != -1) {
  399. $log.warn('ticket already present: not added');
  400. ret = false;
  401. } else {
  402. $scope.event.tickets.push(ticket);
  403. }
  404. // Try to remove this person from the allPersons list using ID of the original entry or email.
  405. var field = null;
  406. var field_value = null;
  407. if (original_ticket && original_ticket._id) {
  408. field = '_id';
  409. field_value = original_ticket._id;
  410. } else if (ticket.email) {
  411. field = 'email';
  412. field_value = ticket.email;
  413. }
  414. if (field) {
  415. var all_person_idx = $scope.allPersons.findIndex(function(el, idx, array) {
  416. return field_value == el[field];
  417. });
  418. if (all_person_idx != -1) {
  419. $scope.allPersons.splice(all_person_idx, 1);
  420. }
  421. }
  422. return ret;
  423. };
  424. $scope._localUpdateTicket = function(ticket) {
  425. if (!$state.is('event.tickets')) {
  426. return;
  427. }
  428. if (!$scope.event.tickets) {
  429. $scope.event.tickets = [];
  430. }
  431. var ticket_idx = $scope.event.tickets.findIndex(function(el, idx, array) {
  432. return ticket._id == el._id;
  433. });
  434. if (ticket_idx == -1) {
  435. $log.warn('ticket not present: not updated');
  436. return false;
  437. }
  438. $scope.event.tickets.splice(ticket_idx, 1, ticket);
  439. };
  440. $scope._localRemoveTicket = function(ticket) {
  441. if (!(ticket && ticket._id && $scope.event.tickets)) {
  442. return;
  443. }
  444. var ticket_idx = $scope.event.tickets.findIndex(function(el, idx, array) {
  445. return ticket._id == el._id;
  446. });
  447. if (ticket_idx == -1) {
  448. $log.warn('unable to find and delete ticket _id ' + ticket._id);
  449. return;
  450. }
  451. var removed_person = $scope.event.tickets.splice(ticket_idx, 1);
  452. // to be used to populate allPersons, if needed.
  453. var person = null;
  454. if (removed_person.length) {
  455. person = removed_person[0];
  456. } else {
  457. return;
  458. }
  459. if (!$scope.allPersons) {
  460. $scope.allPersons = [];
  461. }
  462. var all_person_idx = $scope.allPersons.findIndex(function(el, idx, array) {
  463. return person._id == el._id;
  464. });
  465. if (all_person_idx == -1 && person._id) {
  466. $scope.allPersons.push(person);
  467. }
  468. };
  469. $scope.buildTicketLabel = function(ticket) {
  470. var name = ticket.name || '';
  471. if (ticket.surname) {
  472. if (name) {
  473. name = name + ' ';
  474. }
  475. name = name + ticket.surname;
  476. }
  477. if (!name && ticket.email) {
  478. name = ticket.email;
  479. }
  480. if (!name) {
  481. name = 'ticket';
  482. }
  483. return name;
  484. };
  485. $scope.setTicketAttribute = function(ticket, key, value, callback, hideMessage) {
  486. $log.debug('setTicketAttribute for _id ' + ticket._id + ' key: ' + key + ' value: ' + value);
  487. var newData = {event_id: $state.params.id, _id: ticket._id};
  488. newData[key] = value;
  489. EventTicket.update(newData, function(data) {
  490. if (!(data && data._id && data.ticket)) {
  491. return;
  492. }
  493. if (callback) {
  494. callback(data);
  495. }
  496. if (!$state.is('event.tickets')) {
  497. return;
  498. }
  499. var ticket_idx = $scope.event.tickets.findIndex(function(el, idx, array) {
  500. return data._id == el._id;
  501. });
  502. if (ticket_idx == -1) {
  503. $log.warn('unable to find ticket _id ' + data._id);
  504. return;
  505. }
  506. if ($scope.event.tickets[ticket_idx] != data.ticket) {
  507. $scope.event.tickets.splice(ticket_idx, 1, data.ticket);
  508. }
  509. if (key === 'attended' && !hideMessage) {
  510. $scope.showAttendedMessage(data.ticket, value);
  511. }
  512. });
  513. };
  514. $scope.showAttendedMessage = function(ticket, attends) {
  515. var msg = {};
  516. var msg_type = 'success';
  517. var name = $scope.buildTicketLabel(ticket);
  518. if (attends) {
  519. msg.message = name + ' successfully added to event ' + $scope.event.title;
  520. } else {
  521. msg.message = name + ' successfully removed from event ' + $scope.event.title;
  522. msg_type = 'warning';
  523. }
  524. toaster.pop({type: msg_type, title: msg.message});
  525. };
  526. $scope.setTicketAttributeAndRefocus = function(ticket, key, value) {
  527. $scope.setTicketAttribute(ticket, key, value);
  528. $scope.query = '';
  529. };
  530. $scope._setAttended = function(ticket) {
  531. $scope.setTicketAttribute(ticket, 'attended', true, null, true);
  532. };
  533. $scope.deleteTicket = function(ticket) {
  534. EventTicket.delete({
  535. event_id: $state.params.id,
  536. ticket_id: ticket._id
  537. }, function() {
  538. $scope._localRemoveTicket(ticket);
  539. var msg = $scope.buildTicketLabel(ticket);
  540. msg += ' successfully removed from event ' + $scope.event.title;
  541. toaster.pop({type: 'error', title: msg});
  542. });
  543. };
  544. $scope.addTicket = function(ticket, cb) {
  545. ticket.event_id = $state.params.id;
  546. if ($scope.modalInstance && $scope.modalInstance.opened) {
  547. $scope.modalInstance.close();
  548. }
  549. EventTicket.add(ticket, function(ret_ticket) {
  550. $log.debug('addTicket');
  551. $log.debug(ret_ticket);
  552. $scope.$emit('event:ticket:new', ret_ticket, function() {
  553. $scope.$emit('event:ticket:set-attr', ret_ticket, 'attended', true, null, true);
  554. });
  555. if (cb) {
  556. cb(ticket);
  557. }
  558. if (!$state.is('event.tickets')) {
  559. $state.go('event.ticket.edit', {id: $scope.event._id, ticket_id: ret_ticket._id});
  560. } else {
  561. $scope.query = '';
  562. if ($scope.modalInstance) {
  563. // Close the Quick ticket modal.
  564. $scope.modalInstance.dismiss('no reason');
  565. }
  566. var msg = $scope.buildTicketLabel(ret_ticket);
  567. msg += ' successfully added to event ' + $scope.event.title;
  568. toaster.pop({type: 'success', title: msg});
  569. }
  570. });
  571. };
  572. $scope.updateTicket = function(ticket, cb) {
  573. ticket.event_id = $state.params.id;
  574. EventTicket.update(ticket, function(t) {
  575. $scope.$emit('event:ticket:update', t.ticket);
  576. if (cb) {
  577. cb(t);
  578. }
  579. });
  580. };
  581. $scope.toggleCancelledTicket = function() {
  582. if (!$scope.ticket._id) {
  583. return;
  584. }
  585. $scope.ticket.cancelled = !$scope.ticket.cancelled;
  586. $scope.setTicketAttribute($scope.ticket, 'cancelled', $scope.ticket.cancelled, function() {
  587. $scope.guiOptions.dangerousActionsEnabled = false;
  588. });
  589. };
  590. $scope.openQuickAddTicket = function(_id) {
  591. $scope.modalInstance = $uibModal.open({
  592. templateUrl: 'modal-quick-add-ticket.html',
  593. scope: $scope
  594. });
  595. };
  596. $scope.submitForm = function(dataModelSubmitted) {
  597. $scope.ticket = {_id: $scope.ticket._id};
  598. angular.forEach(dataModelSubmitted, function(value, key) {
  599. key = $scope.formFieldsMap[key] || key;
  600. $scope.ticket[key] = value;
  601. });
  602. if ($state.is('event.ticket.edit')) {
  603. $scope.updateTicket($scope.ticket, function() {
  604. toaster.pop({type: 'info', title: 'ticket successfully updated'});
  605. });
  606. } else {
  607. $scope.addTicket($scope.ticket);
  608. }
  609. };
  610. $scope.cancelForm = function() {
  611. if (!$state.is('event.tickets')) {
  612. $state.go('events');
  613. } else if ($scope.modalInstance) {
  614. $scope.modalInstance.dismiss('no reason');
  615. }
  616. };
  617. $scope.extractFormFields = function(formlyFieldsModel) {
  618. if (!formlyFieldsModel) {
  619. return;
  620. }
  621. angular.forEach(formlyFieldsModel, function(row, idx) {
  622. if (!row.className == 'row') {
  623. return;
  624. }
  625. angular.forEach(row.fieldGroup || [], function(item, idx) {
  626. if (!(item.key && item.templateOptions && item.templateOptions.label)) {
  627. return;
  628. }
  629. var value = item.templateOptions.label.toLowerCase();
  630. $scope.formFieldsMap[item.key] = value;
  631. $scope.formFieldsMapRev[value] = item.key;
  632. });
  633. });
  634. };
  635. $scope.updateOrded = function(key) {
  636. var new_order = [key];
  637. var inv_key;
  638. if (key && key[0] === '-') {
  639. inv_key = key.substring(1);
  640. } else {
  641. inv_key = '-' + key;
  642. }
  643. angular.forEach($scope.ticketsOrder,
  644. function(value, idx) {
  645. if (value !== key && value !== inv_key) {
  646. new_order.push(value);
  647. }
  648. }
  649. );
  650. $scope.ticketsOrder = new_order;
  651. $scope.filterTickets();
  652. };
  653. $scope.updateCSV = function() {
  654. if (!$scope.filteredTickets.length) {
  655. return;
  656. }
  657. try {
  658. var csv = json2csv({data: $scope.filteredTickets});
  659. var blob = new Blob([csv], {type: 'text/csv'});
  660. $scope.downloadURL = (window.URL || window.webkitURL).createObjectURL(blob);
  661. } catch(err) {}
  662. };
  663. $scope.resetInput = function() {
  664. $scope.query = "";
  665. };
  666. $scope.$on('$destroy', function() {
  667. $scope.EventUpdates && $scope.EventUpdates.close();
  668. });
  669. }]
  670. );
  671. eventManControllers.controller('UsersCtrl', ['$scope', '$rootScope', '$state', '$log', 'User', '$uibModal',
  672. function ($scope, $rootScope, $state, $log, User, $uibModal) {
  673. $scope.loginData = {};
  674. $scope.user = {};
  675. $scope.updateUserInfo = {};
  676. $scope.users = [];
  677. $scope.usersOrderProp = 'username';
  678. $scope.ticketsOrderProp = 'title';
  679. $scope.confirm_delete = 'Do you really want to delete this user?';
  680. $rootScope.$on('$translateChangeSuccess', function () {
  681. $translate('Do you really want to delete this user?').then(function (translation) {
  682. $scope.confirm_delete = translation;
  683. });
  684. });
  685. $scope.updateUsersList = function() {
  686. if ($state.is('users')) {
  687. $scope.users = User.all();
  688. }
  689. };
  690. $scope.updateUsersList();
  691. if ($state.is('user.edit') && $state.params.id) {
  692. $scope.user = User.get({id: $state.params.id}, function() {
  693. $scope.updateUserInfo = $scope.user;
  694. $scope.updateUserInfo.isAdmin = $rootScope.hasPermission('admin|all', $scope.updateUserInfo);
  695. });
  696. }
  697. $scope.updateUser = function() {
  698. User.update($scope.updateUserInfo);
  699. };
  700. $scope.deleteUser = function(user_id) {
  701. var modalInstance = $uibModal.open({
  702. scope: $scope,
  703. templateUrl: 'modal-confirm-action.html',
  704. controller: 'ModalConfirmInstanceCtrl',
  705. resolve: {
  706. message: function() { return $scope.confirm_delete; }
  707. }
  708. });
  709. modalInstance.result.then(function() {
  710. User.delete({id: user_id}, $scope.updateUsersList);
  711. });
  712. };
  713. $scope.register = function() {
  714. User.add($scope.newUser, function(data) {
  715. $scope.login($scope.newUser);
  716. });
  717. };
  718. $scope.login = function(loginData) {
  719. if (!loginData) {
  720. loginData = $scope.loginData;
  721. }
  722. User.login(loginData, function(data) {
  723. if (!data.error) {
  724. $rootScope.readInfo(function(info) {
  725. $log.debug('logged in user: ' + $scope.info.user.username);
  726. $rootScope.clearError();
  727. $state.go('events');
  728. });
  729. }
  730. });
  731. };
  732. $scope.logout = function() {
  733. User.logout({}, function(data) {
  734. if (!data.error) {
  735. $rootScope.readInfo(function() {
  736. $log.debug('logged out user');
  737. $state.go('login');
  738. });
  739. }
  740. });
  741. };
  742. }]
  743. );
  744. eventManControllers.controller('FileUploadCtrl', ['$scope', '$log', '$upload', 'EbAPI', 'Event', 'toaster',
  745. function ($scope, $log, $upload, EbAPI, Event, toaster) {
  746. $scope.file = null;
  747. $scope.progress = 0;
  748. $scope.progressbarType = 'warning';
  749. $scope.deduplicate = false;
  750. $scope.targetEvent = null;
  751. $scope.createNewEvent = true;
  752. $scope.ebAPIkey = '';
  753. $scope.ebEventID = '';
  754. $scope.reply = {};
  755. $scope.events = Event.all();
  756. $scope.importRunning = false;
  757. $scope.apiImport = function() {
  758. if (!($scope.ebAPIkey && $scope.ebEventID)) {
  759. $log.warn('missing Eventbrite API key or Event ID');
  760. return;
  761. }
  762. $scope.importRunning = true;
  763. var watingToaster = toaster.pop({type: 'wait', title: 'importing tickets',
  764. body: 'this may take a while...',
  765. timeout: 0, showCloseButton: false,
  766. tapToDismiss: false});
  767. EbAPI.apiImport({
  768. create: $scope.createNewEvent,
  769. eventID: $scope.ebEventID,
  770. targetEvent: $scope.targetEvent,
  771. oauthToken: $scope.ebAPIkey
  772. }, function(data) {
  773. toaster.clear(watingToaster);
  774. toaster.pop({type: 'info', title: 'tickets imported!',
  775. body: 'total: ' + data.total + ' errors: ' + (data.total - data.valid)})
  776. $scope.importRunning = false;
  777. }, function(data) {
  778. toaster.clear(watingToaster);
  779. $scope.importRunning = false;
  780. });
  781. };
  782. $scope.upload = function(file, url) {
  783. $log.debug("FileUploadCtrl.upload");
  784. $scope.progress = 0;
  785. $scope.progressbarType = 'warning';
  786. $upload.upload({
  787. url: url,
  788. file: file,
  789. fields: {targetEvent: $scope.targetEvent, deduplicate: $scope.deduplicate}
  790. }).progress(function(evt) {
  791. $scope.progress = parseInt(100.0 * evt.loaded / evt.total);
  792. $log.debug('progress: ' + $scope.progress + '%');
  793. }).success(function(data, status, headers, config) {
  794. $scope.file = null;
  795. $scope.progress = 100;
  796. $scope.progressbarType = 'success';
  797. $scope.reply = angular.fromJson(data);
  798. });
  799. };
  800. }]
  801. );