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 // Url retrieval function
11 window.baseUrl = function(path) {
12 let basePath = document.querySelector('meta[name="base-url"]').getAttribute('content');
13 if (basePath[basePath.length-1] === '/') basePath = basePath.slice(0, basePath.length-1);
14 if (path[0] === '/') path = path.slice(1);
15 return basePath + '/' + path;
18 var ngApp = angular.module('bookStack', ['ngResource', 'ngAnimate', 'ngSanitize', 'ui.sortable']);
20 // Global Event System
26 emit(eventName, eventData) {
27 if (typeof this.listeners[eventName] === 'undefined') return this;
28 var eventsToStart = this.listeners[eventName];
29 for (let i = 0; i < eventsToStart.length; i++) {
30 var event = eventsToStart[i];
36 listen(eventName, callback) {
37 if (typeof this.listeners[eventName] === 'undefined') this.listeners[eventName] = [];
38 this.listeners[eventName].push(callback);
42 window.Events = new EventManager();
45 var services = require('./services')(ngApp, window.Events);
46 var directives = require('./directives')(ngApp, window.Events);
47 var controllers = require('./controllers')(ngApp, window.Events);
49 //Global jQuery Config & Extensions
52 jQuery.fn.smoothScrollTo = function () {
53 if (this.length === 0) return;
54 let scrollElem = document.documentElement.scrollTop === 0 ? document.body : document.documentElement;
55 $(scrollElem).animate({
56 scrollTop: this.offset().top - 60 // Adjust to change final scroll position top margin
57 }, 800); // Adjust to change animations speed (ms)
61 // Making contains text expression not worry about casing
62 jQuery.expr[":"].contains = $.expr.createPseudo(function (arg) {
63 return function (elem) {
64 return $(elem).text().toUpperCase().indexOf(arg.toUpperCase()) >= 0;
68 // Global jQuery Elements
71 var notifications = $('.notification');
72 var successNotification = notifications.filter('.pos');
73 var errorNotification = notifications.filter('.neg');
74 var warningNotification = notifications.filter('.warning');
75 // Notification Events
76 window.Events.listen('success', function (text) {
77 successNotification.hide();
78 successNotification.find('span').text(text);
80 successNotification.show();
83 window.Events.listen('warning', function (text) {
84 warningNotification.find('span').text(text);
85 warningNotification.show();
87 window.Events.listen('error', function (text) {
88 errorNotification.find('span').text(text);
89 errorNotification.show();
92 // Notification hiding
93 notifications.click(function () {
97 // Chapter page list toggles
98 $('.chapter-toggle').click(function (e) {
100 $(this).toggleClass('open');
101 $(this).closest('.chapter').find('.inset-list').slideToggle(180);
104 // Back to top button
105 $('#back-to-top').click(function() {
106 $('#header').smoothScrollTo();
108 var scrollTopShowing = false;
109 var scrollTop = document.getElementById('back-to-top');
110 var scrollTopBreakpoint = 1200;
111 window.addEventListener('scroll', function() {
112 let scrollTopPos = document.documentElement.scrollTop || document.body.scrollTop || 0;
113 if (!scrollTopShowing && scrollTopPos > scrollTopBreakpoint) {
114 scrollTop.style.display = 'block';
115 scrollTopShowing = true;
117 scrollTop.style.opacity = 0.4;
119 } else if (scrollTopShowing && scrollTopPos < scrollTopBreakpoint) {
120 scrollTop.style.opacity = 0;
121 scrollTopShowing = false;
123 scrollTop.style.display = 'none';
128 // Common jQuery actions
129 $('[data-action="expand-entity-list-details"]').click(function() {
130 $('.entity-list.compact').find('p').not('.empty-text').slideToggle(240);
134 $('.popup-close').click(function() {
135 $(this).closest('.overlay').fadeOut(240);
138 $('.overlay').click(function(event) {
139 if (!$(event.target).hasClass('overlay')) return;
140 $(this).fadeOut(240);
145 // Page specific items
146 require('./pages/page-show');