X-Git-Url: http://source.bookstackapp.com/bookstack/blobdiff_plain/6b4ec65b0303487fc45024dae8b814c054e535ef..refs/pull/696/head:/resources/assets/js/global.js diff --git a/resources/assets/js/global.js b/resources/assets/js/global.js index 66326da23..352616c5a 100644 --- a/resources/assets/js/global.js +++ b/resources/assets/js/global.js @@ -1,51 +1,130 @@ -$(function () { +"use strict"; +require("babel-polyfill"); +require('./dom-polyfills'); - // Notification hiding - $('.notification').click(function () { - $(this).fadeOut(100); +// Url retrieval function +window.baseUrl = function(path) { + let basePath = document.querySelector('meta[name="base-url"]').getAttribute('content'); + if (basePath[basePath.length-1] === '/') basePath = basePath.slice(0, basePath.length-1); + if (path[0] === '/') path = path.slice(1); + return basePath + '/' + path; +}; - }); +// Global Event System +class EventManager { + constructor() { + this.listeners = {}; + this.stack = []; + } + + emit(eventName, eventData) { + this.stack.push({name: eventName, data: eventData}); + if (typeof this.listeners[eventName] === 'undefined') return this; + let eventsToStart = this.listeners[eventName]; + for (let i = 0; i < eventsToStart.length; i++) { + let event = eventsToStart[i]; + event(eventData); + } + return this; + } + + listen(eventName, callback) { + if (typeof this.listeners[eventName] === 'undefined') this.listeners[eventName] = []; + this.listeners[eventName].push(callback); + return this; + } +} - // Dropdown toggles - $('[data-dropdown]').dropDown(); +window.$events = new EventManager(); - // Chapter page list toggles - $('.chapter-toggle').click(function(e) { - e.preventDefault(); - $(this).toggleClass('open'); - $(this).closest('.chapter').find('.inset-list').slideToggle(180); - }); +const Vue = require("vue"); +const axios = require("axios"); +let axiosInstance = axios.create({ + headers: { + 'X-CSRF-TOKEN': document.querySelector('meta[name=token]').getAttribute('content'), + 'baseURL': window.baseUrl('') + } +}); +axiosInstance.interceptors.request.use(resp => { + return resp; +}, err => { + if (typeof err.response === "undefined" || typeof err.response.data === "undefined") return Promise.reject(err); + if (typeof err.response.data.error !== "undefined") window.$events.emit('error', err.response.data.error); + if (typeof err.response.data.message !== "undefined") window.$events.emit('error', err.response.data.message); }); +window.$http = axiosInstance; +Vue.prototype.$http = axiosInstance; +Vue.prototype.$events = window.$events; -// Vue Components +// Translation setup +// Creates a global function with name 'trans' to be used in the same way as Laravel's translation system +const Translations = require("./translations"); +let translator = new Translations(window.translations); +window.trans = translator.get.bind(translator); +window.trans_choice = translator.getPlural.bind(translator); -Vue.component('image-picker', { - template: require('./templates/image-picker.html'), - props: ['currentImage', 'name', 'imageClass'], - data: function() { - return { - image: this.currentImage - } - }, - methods: { - showImageManager: function(e) { - var _this = this; - ImageManager.show(function(image) { - _this.image = image.url; - }); - }, - reset: function() { - this.image = ''; - }, - remove: function() { - this.image = 'none'; - } - } + +require("./vues/vues"); +require("./components"); + + +//Global jQuery Config & Extensions + +/** + * Scroll the view to a specific element. + * @param {HTMLElement} element + */ +window.scrollToElement = function(element) { + if (!element) return; + let offset = window.pageYOffset || document.documentElement.scrollTop || document.body.scrollTop; + let top = element.getBoundingClientRect().top + offset; + $('html, body').animate({ + scrollTop: top - 60 // Adjust to change final scroll position top margin + }, 300); +}; + +/** + * Scroll and highlight an element. + * @param {HTMLElement} element + */ +window.scrollAndHighlight = function(element) { + if (!element) return; + window.scrollToElement(element); + let color = document.getElementById('custom-styles').getAttribute('data-color-light'); + let initColor = window.getComputedStyle(element).getPropertyValue('background-color'); + element.style.backgroundColor = color; + setTimeout(() => { + element.classList.add('selectFade'); + element.style.backgroundColor = initColor; + }, 10); + setTimeout(() => { + element.classList.remove('selectFade'); + element.style.backgroundColor = ''; + }, 3000); +}; + +// Smooth scrolling +jQuery.fn.smoothScrollTo = function () { + if (this.length === 0) return; + window.scrollToElement(this[0]); + return this; +}; + +// Making contains text expression not worry about casing +jQuery.expr[":"].contains = $.expr.createPseudo(function (arg) { + return function (elem) { + return $(elem).text().toUpperCase().indexOf(arg.toUpperCase()) >= 0; + }; }); -// Global Vue Instance -var app = new Vue({ - el: '#app' -}); \ No newline at end of file +// Detect IE for css +if(navigator.userAgent.indexOf('MSIE')!==-1 + || navigator.appVersion.indexOf('Trident/') > 0 + || navigator.userAgent.indexOf('Safari') !== -1){ + document.body.classList.add('flexbox-support'); +} + +// Page specific items +require("./pages/page-show");