/* * Converts JSON data received from the server to flat array of values. */ function jsonToValuesArray(jsonData) { var errMsgs = []; for (var key in jsonData) { var values = jsonData[key]; $.each(values, function (idx, val) { errMsgs.push(val); }); } return errMsgs; } /* * Calls callback function on AJAX success (because built-in functions don't * work!) */ $.fn.onAjaxComplete = function (cb) { $(this) .on('ajax:success', function () { cb(); }) .on('ajax:error', function () { cb(); }); } // Number of all tutorial steps var TUTORIAL_STEPS_CNT = 26; /** * Initializes tutorial steps for the current page. * NOTE: You can specify steps manually in JS with steps parameter (preferred * way), or hardcode them in HTML * NOTE: If some steps edit page, then this function needs to be called several * times for the same page, but for different steps. The same goes if the page * has discontinuous tutorial steps. In such cases, use steps branching, e.g.: * @example * var tutorialData = Cookies.get('tutorial_data'); * if (tutorialData) { * tutorialData = JSON.parse(tutorialData); * var stepNum = parseInt(Cookies.get('current_tutorial_step'), 10); * * if (stepNum >= 6 && stepNum <= 7) { * ... * } else if ... * NOTE: If an element the popup is pointing at is of lesser horizontal length * than the popup itself, then it will not be positioned correctly if it's * position is top or bottom, so set/change the step's position to either left * or right (and don't use any custom styling!), e.g.: * @example * var steps = [ * { * ... * position: 'right' * }, * { * ... * ]; * NOTE: If only one page step is needed, then make pageFirstStepN == * pageLastStepN (both represent the one and only step number) * * @param {number} pageFirstStepN Page's first step number * @param {number} pageLastStepN Page's last step number * @param {string} nextPagePath Next page absolute path * @param {function} beforeCb Callback called before the tutorial starts. Mainly * used for setting 'pointer-events: none' on the elements the page's steps * highlight. * @param {function} endCb Callback called after the tutorial ends. Mainly used * for setting 'pointer-events: auto' on the elements the page's steps * highlight. * @param {object} steps Optional JSON containing introJs steps. They can be * specified here, or hardcoded in HTML. */ function initPageTutorialSteps(pageFirstStepN, pageLastStepN, nextPagePath, beforeCb, endCb, steps) { var tutorialData = Cookies.get('tutorial_data'); if (tutorialData) { tutorialData = JSON.parse(tutorialData); var stepNum = parseInt(Cookies.get('current_tutorial_step'), 10); if (isNaN(stepNum)) { // Cookies data initialization stepNum = 1; Cookies.set('current_tutorial_step', stepNum); tutorialData[0].backPagesPaths = []; Cookies.set('tutorial_data', tutorialData); } var thisPagePath = window.location.pathname; beforeCb(); // Initialize tutorial for the current page's steps var doneLabel = (pageLastStepN === TUTORIAL_STEPS_CNT) ? 'Start using sciNote' : 'End tutorial'; if (_.isUndefined(steps)) { introJs() .setOptions({ overlayOpacity: '0.2', prevLabel: 'Back', nextLabel: 'Next', skipLabel: 'End tutorial', doneLabel: doneLabel, showBullets: false, showStepNumbers: false, exitOnOverlayClick: false, exitOnEsc: false, disableInteraction: true, keyboardNavigation: false, tooltipClass: 'custom next-page-link' }) .goToStep(stepNum - (pageFirstStepN - 1)) .onexit(function() { Cookies.remove('tutorial_data'); Cookies.remove('current_tutorial_step'); location.reload(); }) .oncomplete(function() { Cookies.remove('tutorial_data'); Cookies.remove('current_tutorial_step'); location.reload(); }) .start(); } else { if (pageFirstStepN === pageLastStepN) { // Only one page step, so add another fake one, so the back and next // buttons are added to the popup steps.push({}); } introJs() .setOptions({ overlayOpacity: '0.2', prevLabel: 'Back', nextLabel: 'Next', skipLabel: 'End tutorial', doneLabel: doneLabel, showBullets: false, showStepNumbers: false, exitOnOverlayClick: false, exitOnEsc: false, disableInteraction: true, keyboardNavigation: false, tooltipClass: 'custom next-page-link', steps: steps }) .onexit(function() { location.reload(); }) .oncomplete(function() { location.reload(); }) .goToStep(stepNum - (pageFirstStepN - 1)) .onexit(function() { Cookies.remove('tutorial_data'); Cookies.remove('current_tutorial_step'); }) .oncomplete(function() { Cookies.remove('tutorial_data'); Cookies.remove('current_tutorial_step'); }) .start(); } // Page navigation when coming to this page from previous/next page $(function() { if (stepNum === pageFirstStepN && stepNum > 1) { $('.introjs-prevbutton').removeClass('introjs-disabled'); } else if (stepNum === pageLastStepN && stepNum < TUTORIAL_STEPS_CNT) { $('.introjs-nextbutton').removeClass('introjs-disabled'); } }); // Page navigation when already on this page $('.introjs-skipbutton').click(function() { Cookies.remove('tutorial_data'); Cookies.remove('current_tutorial_step'); endCb(); }); $('.introjs-prevbutton').click(function() { if (stepNum > 1) { Cookies.set('current_tutorial_step', --stepNum); if (stepNum === pageFirstStepN && stepNum > 1) { $('.introjs-prevbutton').removeClass('introjs-disabled'); } else if (stepNum < pageFirstStepN) { // Go to previous page; var prevPagePath = tutorialData[0].backPagesPaths.pop(); Cookies.set('tutorial_data', tutorialData); $('.introjs-prevbutton').attr('href', prevPagePath); introJs().exit(); endCb(); } } }); $('.introjs-nextbutton').click(function() { if (stepNum < TUTORIAL_STEPS_CNT) { Cookies.set('current_tutorial_step', ++stepNum); if (stepNum === pageLastStepN && stepNum < TUTORIAL_STEPS_CNT) { $('.introjs-nextbutton').removeClass('introjs-disabled'); } else if (stepNum > pageLastStepN) { // Go to next page tutorialData[0].backPagesPaths.push(thisPagePath); Cookies.set('tutorial_data', tutorialData); $('.introjs-nextbutton').attr('href', nextPagePath); introJs().exit(); endCb(); } } }); } } /** * Add redirection links on dropdown elements. * @param {[type]} selectedIdx Index of element to be selected. * @return {Object} This */ $.fn.makeDropdownOptionsLinks = function(selectedIdx) { selectedIdx = _.isUndefined(selectedIdx) ? 1 : selectedIdx; $(this).change(function() { location = $(this).find('option:selected').attr('href'); }); $(this).find('option') .each(function(i) { if (!$(this).is("[href]")) { $(this).attr('href', '#'); } }) .eq(selectedIdx).attr('selected', true); return this; }