jQuery.noConflict(); /** * File: LeftAndMain.js */ (function($) { // setup jquery.entwine $.entwine.warningLevel = $.entwine.WARN_LEVEL_BESTPRACTISE; $.entwine('ss', function($) { /** * Position the loading spinner animation below the ss logo */ var positionLoadingSpinner = function() { var offset = 120; // offset from the ss logo var spinner = $('.ss-loading-screen .loading-animation'); var top = ($(window).height() - spinner.height()) / 2; spinner.css('top', top + offset); spinner.show(); }; // apply an select element only when it is ready, ie. when it is rendered into a template // with css applied and got a width value. var applyChosen = function(el){ if(el.outerWidth()){ el.chosen({ 'allow_single_deselect': true }).addClass("has-chzn"); // Copy over title attribute if required if(el.attr('title')) el.siblings('.chzn-container').attr('title', el.attr('title')); } else { setTimeout(function() {applyChosen(el);},500); } }; $(window).bind('resize', positionLoadingSpinner).trigger('resize'); // global ajax handlers $(document).ajaxComplete(function(e, xhr, settings) { // Simulates a redirect on an ajax response. if(window.History.enabled) { var url = xhr.getResponseHeader('X-ControllerURL'), opts, requestHeaders = settings.headers; // Normalize trailing slashes in URL to work around routing weirdnesses in SS_HTTPRequest. var isSame = (url && History.getPageUrl().replace(/\/+$/, '') == url.replace(/\/+$/, '')); if(url && !isSame) { opts = {pjax: xhr.getResponseHeader('X-Pjax') ? xhr.getResponseHeader('X-Pjax') : settings.headers['X-Pjax']}; window.History.pushState(opts, '', url); } } // Handle custom status message headers var msg = (xhr.getResponseHeader('X-Status')) ? xhr.getResponseHeader('X-Status') : xhr.statusText, msgType = (xhr.status < 200 || xhr.status > 399) ? 'bad' : 'good', ignoredMessages = ['OK']; // Show message (but ignore aborted requests) if(xhr.status !== 0 && msg && $.inArray(msg, ignoredMessages)) { // Decode into UTF-8, HTTP headers don't allow multibyte statusMessage(decodeURIComponent(msg), msgType); } }); /** * Main LeftAndMain interface with some control panel and an edit form. * * Events: * ajaxsubmit - ... * validate - ... * aftersubmitform - ... */ $('.cms-container').entwine({ CurrentXHR: null, StateChangeCount: 0, /** * Constructor: onmatch */ onadd: function() { var self = this; // Browser detection if($.browser.msie && parseInt($.browser.version, 10) < 7) { $('.ss-loading-screen').append( '
' + 'Your browser is not compatible with the CMS interface. Please use Internet Explorer 7+, Google Chrome 10+ or Mozilla Firefox 3.5+.' + '
' ).css('z-index', $('.ss-loading-screen').css('z-index')+1); $('.loading-animation').remove(); this._super(); return; } // Initialize layouts this.redraw(); // Remove loading screen $('.ss-loading-screen').hide(); $('body').removeClass('loading'); $(window).unbind('resize', positionLoadingSpinner); this._super(); }, fromWindow: { onstatechange: function(){ this.handleStateChange(); }, onresize: function(){ this.redraw(); } }, 'from .cms-panel': { ontoggle: function(){ this.redraw(); } }, 'from .cms-container': { onaftersubmitform: function(){ this.redraw(); } }, redraw: function() { if(window.debug) console.log('redraw', this.attr('class'), this.get(0)); // Move from inner to outer layouts. Some of the elements might not exist. // Not all edit forms are layouted, so qualify by their data value. this.layout({resize: false}); this.find('.cms-panel-layout').redraw(); this.find('.cms-content-fields[data-layout-type]').redraw(); this.find('.cms-edit-form[data-layout-type]').redraw(); this.find('.cms-preview').redraw(); this.find('.cms-content').redraw(); this.layout({resize: false}); }, /** * Proxy around History.pushState() which handles non-HTML5 fallbacks, * as well as global change tracking. Change tracking needs to be synchronous rather than event/callback * based because the user needs to be able to abort the action completely. * * See handleStateChange() for more details. * * Parameters: * - {String} url * - {String} title New window title * - {Object} data Any additional data passed through to History.pushState() */ loadPanel: function(url, title, data) { if(!data) data = {}; if(!title) title = ""; // Check change tracking (can't use events as we need a way to cancel the current state change) var contentEls = this._findFragments(data.pjax ? data.pjax.split(',') : ['Content']); var trackedEls = contentEls.find(':data(changetracker)').add(contentEls.filter(':data(changetracker)')); if(trackedEls.length) { var abort = false; trackedEls.each(function() { if(!$(this).confirmUnsavedChanges()) abort = true; }); if(abort) return; } if(window.History.enabled) { // Active menu item is set based on X-Controller ajax header, // which matches one class on the menu window.History.pushState(data, title, url); } else { window.location = $.path.makeUrlAbsolute(url, $('base').attr('href')); } }, /** * Function: submitForm * * Parameters: * {DOMElement} form - The form to be submitted. Needs to be passed * in to avoid entwine methods/context being removed through replacing the node itself. * {DOMElement} button - The pressed button (optional) * {Function} callback - Called in complete() handler of jQuery.ajax() * {Object} ajaxOptions - Object literal to merge into $.ajax() call * * Returns: * (boolean) */ submitForm: function(form, button, callback, ajaxOptions) { var self = this; // look for save button if(!button) button = this.find('.Actions :submit[name=action_save]'); // default to first button if none given - simulates browser behaviour if(!button) button = this.find('.Actions :submit:first'); form.trigger('beforesubmitform'); this.trigger('submitform', {form: form, button: button}); // set button to "submitting" state $(button).addClass('loading'); // validate if required if(!form.validate()) { // TODO Automatically switch to the tab/position of the first error statusMessage("Validation failed.", "bad"); $(button).removeClass('loading'); return false; } // save tab selections in order to reconstruct them later var selectedTabs = []; form.find('.cms-tabset').each(function(i, el) { if($(el).attr('id')) selectedTabs.push({id:$(el).attr('id'), selected:$(el).tabs('option', 'selected')}); }); // get all data from the form var formData = form.serializeArray(); // add button action formData.push({name: $(button).attr('name'), value:'1'}); // Artificial HTTP referer, IE doesn't submit them via ajax. // Also rewrites anchors to their page counterparts, which is important // as automatic browser ajax response redirects seem to discard the hash/fragment. formData.push({name: 'BackURL', value:History.getPageUrl()}); // Standard Pjax behaviour is to replace the submitted form with new content. // The returned view isn't always decided upon when the request // is fired, so the server might decide to change it based on its own logic, // sending back different `X-Pjax` headers and content jQuery.ajax(jQuery.extend({ headers: {"X-Pjax" : "CurrentForm,Breadcrumbs"}, url: form.attr('action'), data: formData, type: 'POST', complete: function() { $(button).removeClass('loading'); }, success: function(data, status, xhr) { form.removeClass('changed'); // TODO This should be using the plugin API if(callback) callback(data, status, xhr); var newContentEls = self.handleAjaxResponse(data, status, xhr); if(!newContentEls) return; var newForm = newContentEls.filter('form'); // Re-init tabs (in case the form tag itself is a tabset) if(newForm.hasClass('cms-tabset')) newForm.removeClass('cms-tabset').addClass('cms-tabset'); // re-select previously saved tabs $.each(selectedTabs, function(i, selectedTab) { newForm.find('#' + selectedTab.id).tabs('select', selectedTab.selected); }); newForm.trigger('aftersubmitform', {status: status, xhr: xhr, formData: formData}); } }, ajaxOptions)); return false; }, /** * Handles ajax loading of new panels through the window.History object. * To trigger loading, pass a new URL to window.History.pushState(). * Use loadPanel() as a pushState() wrapper as it provides some additional functionality * like global changetracking and user aborts. * * Due to the nature of history management, no callbacks are allowed. * Use the 'beforestatechange' and 'afterstatechange' events instead, * or overwrite the beforeLoad() and afterLoad() methods on the * DOM element you're loading the new content into. * Although you can pass data into pushState(), it shouldn't contain * DOM elements or callback closures. * * The passed URL should allow reconstructing important interface state * without additional parameters, in the following use cases: * - Explicit loading through History.pushState() * - Implicit loading through browser navigation event triggered by the user (forward or back) * - Full window refresh without ajax * For example, a ModelAdmin search event should contain the search terms * as URL parameters, and the result display should automatically appear * if the URL is loaded without ajax. */ handleStateChange: function() { // Don't allow parallel loading to avoid edge cases if(this.getCurrentXHR()) this.getCurrentXHR().abort(); var self = this, h = window.History, state = h.getState(), fragments = state.data.pjax || 'Content', headers = {}, fragmentsArr = fragments.split(','), contentEls = this._findFragments(fragmentsArr); // For legacy IE versions (IE7 and IE8), reload without ajax // as a crude way to fix memory leaks through whole window refreshes. this.setStateChangeCount(this.getStateChangeCount() + 1); var isLegacyIE = ($.browser.msie && parseInt($.browser.version, 10) < 9); if(isLegacyIE && this.getStateChangeCount() > 20) { document.location.href = state.url; return; } // If any of the requested Pjax fragments don't exist in the current view, // fetch the "Content" view instead, which is the "outermost" fragment // that can be reloaded without reloading the whole window. if(contentEls.length < fragmentsArr.length) { fragments = 'Content', fragmentsArr = ['Content']; contentEls = this._findFragments(fragmentsArr); } this.trigger('beforestatechange', {state: state, element: contentEls}); // Set Pjax headers, which can declare a preference for the returned view. // The actually returned view isn't always decided upon when the request // is fired, so the server might decide to change it based on its own logic. headers['X-Pjax'] = fragments; contentEls.addClass('loading'); var xhr = $.ajax({ headers: headers, url: state.url, complete: function() { // Remove loading indication from old content els (regardless of which are replaced) contentEls.removeClass('loading'); }, success: function(data, status, xhr) { var els = self.handleAjaxResponse(data, status, xhr); self.trigger('afterstatechange', {data: data, status: status, xhr: xhr, element: els}); } }); this.setCurrentXHR(xhr); }, /** * Handles ajax responses containing plain HTML, or mulitple * PJAX fragments wrapped in JSON (see PjaxResponseNegotiator PHP class). * Can be hooked into an ajax 'success' callback. */ handleAjaxResponse: function(data, status, xhr) { var self = this; // Pseudo-redirects via X-ControllerURL might return empty data, in which // case we'll ignore the response if(!data) return; // Update title var title = xhr.getResponseHeader('X-Title'); if(title) document.title = title; var newFragments = {}, newContentEls; if(xhr.getResponseHeader('Content-Type') == 'text/json') { newFragments = data; } else { // Fall back to replacing the content fragment if HTML is returned newFragments['Content'] = data; } // Replace each fragment individually $.each(newFragments, function(newFragment, html) { var contentEl = $('[data-pjax-fragment]').filter(function() { return $.inArray(newFragment, $(this).data('pjaxFragment').split(' ')) != -1; }), newContentEl = $(html); // Add to result collection if(newContentEls) newContentEls.add(newContentEl); else newContentEls = newContentEl; // Update panels if(newContentEl.find('.cms-container').length) { throw 'Content loaded via ajax is not allowed to contain tags matching the ".cms-container" selector to avoid infinite loops'; } // Set loading state and store element state var origStyle = contentEl.attr('style'); var origVisible = contentEl.is(':visible'); var layoutClasses = ['east', 'west', 'center', 'north', 'south']; var elemClasses = contentEl.attr('class'); var origLayoutClasses = []; if(elemClasses) { origLayoutClasses = $.grep( elemClasses.split(' '), function(val) { return ($.inArray(val, layoutClasses) >= 0);} ); } newContentEl .removeClass(layoutClasses.join(' ')) .addClass(origLayoutClasses.join(' ')); if(origStyle) newContentEl.attr('style', origStyle); newContentEl.css('visibility', 'hidden'); // Allow injection of inline styles, as they're not allowed in the document body. // Not handling this through jQuery.ondemand to avoid parsing the DOM twice. var styles = newContentEl.find('style').detach(); if(styles.length) $(document).find('head').append(styles); // Replace panel completely (we need to override the "layout" attribute, so can't replace the child instead) contentEl.replaceWith(newContentEl); // Unset loading and restore element state (to avoid breaking existing panel visibility, e.g. with preview expanded) if(origVisible) newContentEl.css('visibility', 'visible'); }); this.redraw(); return newContentEls; }, /** * * * Parameters: * - fragments {Array} * Returns: jQuery collection */ _findFragments: function(fragments) { return $('[data-pjax-fragment]').filter(function() { // Allows for more than one fragment per node var i, nodeFragments = $(this).data('pjaxFragment').split(' '); for(i in fragments) { if($.inArray(fragments[i], nodeFragments) != -1) return true; } return false; }); }, /** * Function: refresh * * Updates the container based on the current url * * Returns: void */ refresh: function() { $(window).trigger('statechange'); $(this).redraw(); } }); /** * Add loading overlay to selected regions in the CMS automatically. * Not applied to all "*.loading" elements to avoid secondary regions * like the breadcrumbs showing unnecessary loading status. */ $('form.loading,.cms-content.loading,.cms-content-fields.loading,.cms-content-view.loading').entwine({ onmatch: function() { this.append(''); this._super(); }, onunmatch: function() { this.find('.cms-content-loading-overlay,.cms-content-loading-spinner').remove(); this._super(); } }); /** Make all buttons "hoverable" with jQuery theming. */ $('.cms input[type="submit"], .cms button, .cms input[type="reset"], .cms .ss-ui-button').entwine({ onadd: function() { this.addClass('ss-ui-button'); if(!this.data('button')) this.button(); this._super(); }, onremove: function() { this.button('destroy'); this._super(); } }); /** * Loads the link's 'href' attribute into a panel via ajax, * as opposed to triggering a full page reload. * Little helper to avoid repetition, and make it easy to * "opt in" to panel loading, while by default links still exhibit their default behaviour. * The PJAX target can be specified via a 'data-pjax-target' attribute. */ $('.cms .cms-panel-link').entwine({ onclick: function(e) { var href = this.attr('href'), url = (href && !href.match(/^#/)) ? href : this.data('href'), data = {pjax: this.data('pjaxTarget')}; $('.cms-container').loadPanel(url, null, data); e.preventDefault(); } }); /** * Does an ajax loads of the link's 'href' attribute via ajax and displays any FormResponse messages from the CMS. * Little helper to avoid repetition, and make it easy to trigger actions via a link, * without reloading the page, changing the URL, or loading in any new panel content. */ $('.cms .ss-ui-button-ajax').entwine({ onclick: function(e) { $(this).removeClass('ui-button-text-only'); $(this).addClass('ss-ui-button-loading ui-button-text-icons'); var loading = $(this).find(".ss-ui-loading-icon"); if(loading.length < 1) { loading = $("").addClass('ss-ui-loading-icon ui-button-icon-primary ui-icon'); $(this).prepend(loading); } loading.show(); var href = this.attr('href'), url = href ? href : this.data('href'); jQuery.ajax({ url: url, // Ensure that form view is loaded (rather than whole "Content" template) complete: function(xmlhttp, status) { var msg = (xmlhttp.getResponseHeader('X-Status')) ? xmlhttp.getResponseHeader('X-Status') : xmlhttp.responseText; try { if (typeof msg != "undefined" && msg != null) eval(msg); } catch(e) {} loading.hide(); $(".cms-container").refresh(); $(this).removeClass('ss-ui-button-loading ui-button-text-icons'); $(this).addClass('ui-button-text-only'); }, dataType: 'html' }); e.preventDefault(); } }); /** * Trigger dialogs with iframe based on the links href attribute (see ssui-core.js). */ $('.cms .ss-ui-dialog-link').entwine({ UUID: null, onmatch: function() { this._super(); this.setUUID(new Date().getTime()); }, onunmatch: function() { this._super(); }, onclick: function() { this._super(); var self = this, id = 'ss-ui-dialog-' + this.getUUID(); var dialog = $('#' + id); if(!dialog.length) { dialog = $(''); $('body').append(dialog); } var extraClass = this.data('popupclass')?this.data('popupclass'):''; dialog.ssdialog({iframeUrl: this.attr('href'), autoOpen: true, dialogExtraClass: extraClass}); return false; } }); /** * Add styling to all contained buttons, and create buttonsets if required. */ $('.cms-content .Actions').entwine({ onmatch: function() { this.find('.ss-ui-button').click(function() { var form = this.form; // forms don't natively store the button they've been triggered with if(form) { form.clickedButton = this; // Reset the clicked button shortly after the onsubmit handlers // have fired on the form setTimeout(function() {form.clickedButton = null;}, 10); } }); this.redraw(); this._super(); }, onunmatch: function() { this._super(); }, redraw: function() { if(window.debug) console.log('redraw', this.attr('class'), this.get(0)); // Remove whitespace to avoid gaps with inline elements this.contents().filter(function() { return (this.nodeType == 3 && !/\S/.test(this.nodeValue)); }).remove(); // Init buttons if required this.find('.ss-ui-button').each(function() { if(!$(this).data('button')) $(this).button(); }); // Mark up buttonsets this.find('.ss-ui-buttonset').buttonset(); } }); /** * Duplicates functionality in DateField.js, but due to using entwine we can match * the DOM element on creation, rather than onclick - which allows us to decorate * the field with a calendar icon */ $('.cms .field.date input.text').entwine({ onmatch: function() { var holder = $(this).parents('.field.date:first'), config = holder.data(); if(!config.showcalendar) { this._super(); return; } config.showOn = 'button'; if(config.locale && $.datepicker.regional[config.locale]) { config = $.extend(config, $.datepicker.regional[config.locale], {}); } $(this).datepicker(config); // // Unfortunately jQuery UI only allows configuration of icon images, not sprites // this.next('button').button('option', 'icons', {primary : 'ui-icon-calendar'}); this._super(); }, onunmatch: function() { this._super(); } }); /** * Styled dropdown select fields via chosen. Allows things like search and optgroup * selection support. Rather than manually adding classes to selects we want * styled, we style everything but the ones we tell it not to. * * For the CMS we also need to tell the parent div that his has a select so * we can fix the height cropping. */ $('.cms .field.dropdown select, .cms .field select[multiple]').entwine({ onmatch: function() { if(this.is('.no-chzn')) { this._super(); return; } // Explicitly disable default placeholder if no custom one is defined if(!this.data('placeholder')) this.data('placeholder', ' '); // We could've gotten stale classes and DOM elements from deferred cache. this.removeClass('has-chzn chzn-done'); this.siblings('.chzn-container').remove(); // Apply Chosen applyChosen(this); this._super(); }, onunmatch: function() { this._super(); } }); $(".cms-panel-layout").entwine({ redraw: function() { if(window.debug) console.log('redraw', this.attr('class'), this.get(0)); this.layout({resize: false}); } }); /** * Overload the default GridField behaviour (open a new URL in the browser) * with the CMS-specific ajax loading. */ $('.cms .ss-gridfield').entwine({ showDetailView: function(url) { // Include any GET parameters from the current URL, as the view state might depend on it. // For example, a list prefiltered through external search criteria might be passed to GridField. url = $.path.addSearchParams(url, window.location.search.replace(/^\?/, '')); $('.cms-container').loadPanel(url); } }); /** * Generic search form in the CMS, often hooked up to a GridField results display. */ $('.cms-search-form').entwine({ onsubmit: function() { // Remove empty elements and make the URL prettier var nonEmptyInputs = this.find(':input:not(:submit)').filter(function() { // Use fieldValue() from jQuery.form plugin rather than jQuery.val(), // as it handles checkbox values more consistently var vals = $.grep($(this).fieldValue(), function(val) { return (val);}); return (vals.length); }); var url = this.attr('action'); if(nonEmptyInputs.length) url = $.path.addSearchParams(url, nonEmptyInputs.serialize()); var container = this.closest('.cms-container'); container.find('.cms-edit-form').tabs('select',0); //always switch to the first tab (list view) when searching container.loadPanel(url); return false; }, /** * Resets are processed on the serverside, so need to trigger a submit. */ onreset: function(e) { this.clearForm(); this.submit(); } }); /** * Simple toggle link, which points to a DOm element by its ID selector * in the href attribute (which doubles as an anchor link to that element). */ $('.cms .cms-help-toggle').entwine({ onmatch: function() { this._super(); $(this.attr('href')).hide(); }, onunmatch: function() { this._super(); }, onclick: function(e) { $(this.attr('href')).toggle(); e.preventDefault(); } }); /** * Allows to lazy load a panel, by leaving it empty * and declaring a URL to load its content via a 'url' HTML5 data attribute. * The loaded HTML is cached, with cache key being the 'url' attribute. * In order for this to work consistently, we assume that the responses are stateless. * To avoid caching, add a 'deferred-no-cache' to the node. */ window._panelDeferredCache = {}; $('.cms-panel-deferred').entwine({ onadd: function() { this._super(); this.redraw(); }, onremove: function() { if(window.debug) console.log('saving', this.data('url'), this); // Save the HTML state at the last possible moment. // Don't store the DOM to avoid memory leaks. if(!this.data('deferredNoCache')) window._panelDeferredCache[this.data('url')] = this.html(); this._super(); }, redraw: function() { if(window.debug) console.log('redraw', this.attr('class'), this.get(0)); var self = this, url = this.data('url'); if(!url) throw 'Elements of class .cms-panel-deferred need a "data-url" attribute'; this._super(); // If the node is empty, try to either load it from cache or via ajax. if(!this.children().length) { if(!this.data('deferredNoCache') && typeof window._panelDeferredCache[url] !== 'undefined') { this.html(window._panelDeferredCache[url]); } else { this.addClass('loading'); $.ajax({ url: url, complete: function() { self.removeClass('loading'); }, success: function(data, status, xhr) { self.html(data); } }); } } } }); /** * Lightweight wrapper around jQuery UI tabs. * Ensures that anchor links are set properly, * and any nested tabs are scrolled if they have * their height explicitly set. This is important * for forms inside the CMS layout. */ $('.cms-tabset').entwine({ onadd: function() { // Can't name redraw() as it clashes with other CMS entwine classes this.redrawTabs(); this._super(); }, onremove: function() { this.tabs('destroy'); this._super(); }, redrawTabs: function() { this.rewriteHashlinks(); var id = this.attr('id'), cookieId = 'ui-tabs-' + id, selectedTab = this.find('ul:first .ui-tabs-selected'); // Fix for wrong cookie storage of deselected tabs if($.cookie && id && $.cookie(cookieId) == -1) $.cookie(cookieId, 0); if(!this.data('tabs')) this.tabs({ cookie: ($.cookie && id) ? { expires: 30, path: '/', name: cookieId } : false, ajaxOptions: { // Overwrite ajax loading to use CMS logic instead beforeSend: function(xhr, settings) { var makeAbs = $.path.makeUrlAbsolute, baseUrl = $('base').attr('href'), isSame = (makeAbs(settings.url, baseUrl) == makeAbs(document.location.href)); if(!isSame) $('.cms-container').loadPanel(settings.url); return false; } }, selected: (selectedTab.index() != -1) ? selectedTab.index() : 0 }); }, /** * Replace prefixes for all hashlinks in tabs. * SSViewer rewrites them from "#Root_MyTab" to * e.g. "/admin/#Root_MyTab" which makes them * unusable for jQuery UI. */ rewriteHashlinks: function() { $(this).find('ul a').each(function() { var href = $(this).attr('href').replace(/.*(#.*)/, '$1'); if(href) $(this).attr('href', href); }); } }); }); }(jQuery)); var statusMessage = function(text, type) { jQuery.noticeAdd({text: text, type: type}); }; var errorMessage = function(text) { jQuery.noticeAdd({text: text, type: 'error'}); };