3 // AngularJS - Create application and load components
4 var angular = require('angular');
5 var ngResource = require('angular-resource');
6 var ngAnimate = require('angular-animate');
7 var ngSanitize = require('angular-sanitize');
8 require('angular-ui-sortable');
10 var ngApp = angular.module('bookStack', ['ngResource', 'ngAnimate', 'ngSanitize', 'ui.sortable']);
12 // Global Event System
18 emit(eventName, eventData) {
19 if (typeof this.listeners[eventName] === 'undefined') return this;
20 var eventsToStart = this.listeners[eventName];
21 for (let i = 0; i < eventsToStart.length; i++) {
22 var event = eventsToStart[i];
28 listen(eventName, callback) {
29 if (typeof this.listeners[eventName] === 'undefined') this.listeners[eventName] = [];
30 this.listeners[eventName].push(callback);
34 window.Events = new Events();
36 var services = require('./services')(ngApp, Events);
37 var directives = require('./directives')(ngApp, Events);
38 var controllers = require('./controllers')(ngApp, Events);
40 //Global jQuery Config & Extensions
43 jQuery.fn.smoothScrollTo = function () {
44 if (this.length === 0) return;
45 let scrollElem = document.documentElement.scrollTop === 0 ? document.body : document.documentElement;
46 $(scrollElem).animate({
47 scrollTop: this.offset().top - 60 // Adjust to change final scroll position top margin
48 }, 800); // Adjust to change animations speed (ms)
52 // Making contains text expression not worry about casing
53 jQuery.expr[":"].contains = $.expr.createPseudo(function (arg) {
54 return function (elem) {
55 return $(elem).text().toUpperCase().indexOf(arg.toUpperCase()) >= 0;
59 // Global jQuery Elements
62 var notifications = $('.notification');
63 var successNotification = notifications.filter('.pos');
64 var errorNotification = notifications.filter('.neg');
65 var warningNotification = notifications.filter('.warning');
66 // Notification Events
67 window.Events.listen('success', function (text) {
68 successNotification.hide();
69 successNotification.find('span').text(text);
71 successNotification.show();
74 window.Events.listen('warning', function (text) {
75 warningNotification.find('span').text(text);
76 warningNotification.show();
78 window.Events.listen('error', function (text) {
79 errorNotification.find('span').text(text);
80 errorNotification.show();
83 // Notification hiding
84 notifications.click(function () {
88 // Chapter page list toggles
89 $('.chapter-toggle').click(function (e) {
91 $(this).toggleClass('open');
92 $(this).closest('.chapter').find('.inset-list').slideToggle(180);
96 $('#back-to-top').click(function() {
97 $('#header').smoothScrollTo();
99 var scrollTopShowing = false;
100 var scrollTop = document.getElementById('back-to-top');
101 var scrollTopBreakpoint = 1200;
102 window.addEventListener('scroll', function() {
103 let scrollTopPos = document.documentElement.scrollTop || document.body.scrollTop || 0;
104 if (!scrollTopShowing && scrollTopPos > scrollTopBreakpoint) {
105 scrollTop.style.display = 'block';
106 scrollTopShowing = true;
108 scrollTop.style.opacity = 0.4;
110 } else if (scrollTopShowing && scrollTopPos < scrollTopBreakpoint) {
111 scrollTop.style.opacity = 0;
112 scrollTopShowing = false;
114 scrollTop.style.display = 'none';
119 // Common jQuery actions
120 $('[data-action="expand-entity-list-details"]').click(function() {
121 $('.entity-list.compact').find('p').not('.empty-text').slideToggle(240);
127 // Page specific items
128 require('./pages/page-show');