added hooks and dependencies for tutorial flow
This commit is contained in:
parent
e78726caa4
commit
a342f7e6b1
@ -11,7 +11,7 @@
|
||||
|
||||
- if can? :create, Page
|
||||
- content_for :buttons do
|
||||
= admin_button_tag :new, new_admin_page_url, :class => 'new'
|
||||
= admin_button_tag :new, new_admin_page_url, :class => 'new', :id => "newpage"
|
||||
|
||||
%p!= t('.help')
|
||||
|
||||
|
@ -9,6 +9,8 @@
|
||||
%span= '|'
|
||||
= link_to t('.switch'), '#', :id => 'sites-picker-link'
|
||||
%span= '|'
|
||||
= link_to 'Help', '#guider=help', :id => "tutorial"
|
||||
%span= '|'
|
||||
= link_to t('.logout'), destroy_admin_session_url, :confirm => t('admin.messages.confirm')
|
||||
|
||||
|
||||
|
322
public/javascripts/admin/guiders-1.1.0.js
Normal file
322
public/javascripts/admin/guiders-1.1.0.js
Normal file
@ -0,0 +1,322 @@
|
||||
/**
|
||||
* guiders.js
|
||||
*
|
||||
* version 1.1.0
|
||||
*
|
||||
* Developed at Optimizely. (www.optimizely.com)
|
||||
* We make A/B testing you'll actually use.
|
||||
*
|
||||
* Released under the Apache License 2.0.
|
||||
* www.apache.org/licenses/LICENSE-2.0.html
|
||||
*
|
||||
* Questions about Guiders or Optimizely?
|
||||
* Email us at jeff+pickhardt@optimizely.com or hello@optimizely.com.
|
||||
*
|
||||
* Enjoy!
|
||||
*/
|
||||
|
||||
var guiders = (function($){
|
||||
var guiders = {
|
||||
version: "1.1.0",
|
||||
|
||||
_defaultSettings: {
|
||||
attachTo: null,
|
||||
buttons: [{name: "Close"}],
|
||||
buttonCustomHTML: "",
|
||||
description: "Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.",
|
||||
isHashable: true,
|
||||
onShow: null,
|
||||
overlay: false,
|
||||
position: 0, // 1-12 follows an analog clock, 0 means centered
|
||||
title: "Sample title goes here",
|
||||
width: 400
|
||||
},
|
||||
|
||||
_htmlSkeleton: [
|
||||
"<div class='guider'>",
|
||||
" <div class='guider_content'>",
|
||||
" <h1 class='guider_title'></h1>",
|
||||
" <p class='guider_description'></p>",
|
||||
" <div class='guider_buttons'>",
|
||||
" </div>",
|
||||
" </div>",
|
||||
" <div class='guider_arrow'>",
|
||||
" </div>",
|
||||
"</div>"
|
||||
].join(""),
|
||||
|
||||
_arrowSize: 42, // = arrow's width and height
|
||||
_guiders: {},
|
||||
_currentGuiderID: null,
|
||||
_lastCreatedGuiderID: null,
|
||||
|
||||
_addButtons: function(myGuider) {
|
||||
// Add buttons
|
||||
var guiderButtonsContainer = myGuider.elem.find(".guider_buttons");
|
||||
for (var i = myGuider.buttons.length-1; i >= 0; i--) {
|
||||
var thisButton = myGuider.buttons[i];
|
||||
var thisButtonElem = $("<a></a>", {
|
||||
"class" : "guider_button",
|
||||
"text" : thisButton.name });
|
||||
if (typeof thisButton.classString !== "undefined" && thisButton.classString !== null) {
|
||||
thisButtonElem.addClass(thisButton.classString);
|
||||
}
|
||||
|
||||
guiderButtonsContainer.append(thisButtonElem);
|
||||
|
||||
if (thisButton.onclick) {
|
||||
thisButtonElem.bind("click", thisButton.onclick);
|
||||
} else if (!thisButton.onclick && thisButton.name.toLowerCase() === "close") {
|
||||
thisButtonElem.bind("click", function() { guiders.hideAll(); });
|
||||
} else if (!thisButton.onclick && thisButton.name.toLowerCase() === "next") {
|
||||
thisButtonElem.bind("click", function() { guiders.next(); });
|
||||
}
|
||||
}
|
||||
|
||||
if (myGuider.buttonCustomHTML !== "") {
|
||||
var myCustomHTML = $(myGuider.buttonCustomHTML);
|
||||
myGuider.elem.find(".guider_buttons").append(myCustomHTML);
|
||||
}
|
||||
},
|
||||
|
||||
_attach: function(myGuider) {
|
||||
if (typeof myGuider.attachTo === "undefined" || myGuider === null) {
|
||||
return;
|
||||
}
|
||||
|
||||
var myHeight = myGuider.elem.innerHeight();
|
||||
var myWidth = myGuider.elem.innerWidth();
|
||||
|
||||
if (myGuider.position === 0) {
|
||||
myGuider.elem.css("position", "absolute");
|
||||
myGuider.elem.css("top", ($(window).height() - myHeight) / 3 + $(window).scrollTop() + "px");
|
||||
myGuider.elem.css("left", ($(window).width() - myWidth) / 2 + $(window).scrollLeft() + "px");
|
||||
return;
|
||||
}
|
||||
|
||||
myGuider.attachTo = $(myGuider.attachTo);
|
||||
var base = myGuider.attachTo.offset();
|
||||
var attachToHeight = myGuider.attachTo.innerHeight();
|
||||
var attachToWidth = myGuider.attachTo.innerWidth();
|
||||
|
||||
var top = base.top;
|
||||
var left = base.left;
|
||||
|
||||
var bufferOffset = 0.9 * guiders._arrowSize;
|
||||
|
||||
var offsetMap = { // Follows the form: [height, width]
|
||||
1: [-bufferOffset - myHeight, attachToWidth - myWidth],
|
||||
2: [0, bufferOffset + attachToWidth],
|
||||
3: [attachToHeight/2 - myHeight/2, bufferOffset + attachToWidth],
|
||||
4: [attachToHeight - myHeight, bufferOffset + attachToWidth],
|
||||
5: [bufferOffset + attachToHeight, attachToWidth - myWidth],
|
||||
6: [bufferOffset + attachToHeight, attachToWidth/2 - myWidth/2],
|
||||
7: [bufferOffset + attachToHeight, 0],
|
||||
8: [attachToHeight - myHeight, -myWidth - bufferOffset],
|
||||
9: [attachToHeight/2 - myHeight/2, -myWidth - bufferOffset],
|
||||
10: [0, -myWidth - bufferOffset],
|
||||
11: [-bufferOffset - myHeight, 0],
|
||||
12: [-bufferOffset - myHeight, attachToWidth/2 - myWidth/2]
|
||||
};
|
||||
|
||||
offset = offsetMap[myGuider.position];
|
||||
top += offset[0];
|
||||
left += offset[1];
|
||||
|
||||
myGuider.elem.css({
|
||||
"position":"absolute",
|
||||
"top": top,
|
||||
"left": left
|
||||
});
|
||||
},
|
||||
|
||||
_guiderById: function(id) {
|
||||
if (typeof guiders._guiders[id] === "undefined") {
|
||||
throw "Cannot find guider with id " + id;
|
||||
}
|
||||
return guiders._guiders[id];
|
||||
},
|
||||
|
||||
_showOverlay: function() {
|
||||
$("#guider_overlay").fadeIn("fast");
|
||||
},
|
||||
|
||||
_hideOverlay: function() {
|
||||
$("#guider_overlay").fadeOut("fast");
|
||||
},
|
||||
|
||||
_initializeOverlay: function() {
|
||||
if ($("#guider_overlay").length === 0) {
|
||||
$("<div id=\"guider_overlay\"></div>").hide().appendTo("body");
|
||||
}
|
||||
},
|
||||
|
||||
_styleArrow: function(myGuider) {
|
||||
var position = myGuider.position || 0;
|
||||
if (!position) {
|
||||
return;
|
||||
}
|
||||
var myGuiderArrow = $(myGuider.elem.find(".guider_arrow"));
|
||||
var newClass = {
|
||||
1: "guider_arrow_down",
|
||||
2: "guider_arrow_left",
|
||||
3: "guider_arrow_left",
|
||||
4: "guider_arrow_left",
|
||||
5: "guider_arrow_up",
|
||||
6: "guider_arrow_up",
|
||||
7: "guider_arrow_up",
|
||||
8: "guider_arrow_right",
|
||||
9: "guider_arrow_right",
|
||||
10: "guider_arrow_right",
|
||||
11: "guider_arrow_down",
|
||||
12: "guider_arrow_down"
|
||||
};
|
||||
myGuiderArrow.addClass(newClass[position]);
|
||||
|
||||
var myHeight = myGuider.elem.innerHeight();
|
||||
var myWidth = myGuider.elem.innerWidth();
|
||||
var arrowOffset = guiders._arrowSize / 2;
|
||||
var positionMap = {
|
||||
1: ["right", arrowOffset],
|
||||
2: ["top", arrowOffset],
|
||||
3: ["top", myHeight/2 - arrowOffset],
|
||||
4: ["bottom", arrowOffset],
|
||||
5: ["right", arrowOffset],
|
||||
6: ["left", myWidth/2 - arrowOffset],
|
||||
7: ["left", arrowOffset],
|
||||
8: ["bottom", arrowOffset],
|
||||
9: ["top", myHeight/2 - arrowOffset],
|
||||
10: ["top", arrowOffset],
|
||||
11: ["left", arrowOffset],
|
||||
12: ["left", myWidth/2 - arrowOffset]
|
||||
};
|
||||
var position = positionMap[myGuider.position];
|
||||
myGuiderArrow.css(position[0], position[1] + "px");
|
||||
},
|
||||
|
||||
/**
|
||||
* One way to show a guider to new users is to direct new users to a URL such as
|
||||
* http://www.mysite.com/myapp#guider=welcome
|
||||
*
|
||||
* This can also be used to run guiders on multiple pages, by redirecting from
|
||||
* one page to another, with the guider id in the hash tag.
|
||||
*
|
||||
* Alternatively, if you use a session variable or flash messages after sign up,
|
||||
* you can add selectively add JavaScript to the page: "guiders.show('first');"
|
||||
*/
|
||||
_showIfHashed: function(myGuider) {
|
||||
var GUIDER_HASH_TAG = "guider=";
|
||||
var hashIndex = window.location.hash.indexOf(GUIDER_HASH_TAG);
|
||||
if (hashIndex !== -1) {
|
||||
var hashGuiderId = window.location.hash.substr(hashIndex + GUIDER_HASH_TAG.length);
|
||||
if (myGuider.id.toLowerCase() === hashGuiderId.toLowerCase()) {
|
||||
// Success!
|
||||
guiders.show(myGuider.id);
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
next: function() {
|
||||
var currentGuider = guiders._guiders[guiders._currentGuiderID];
|
||||
if (typeof currentGuider === "undefined") {
|
||||
return;
|
||||
}
|
||||
var nextGuiderId = currentGuider.next || null;
|
||||
if (nextGuiderId !== null && nextGuiderId !== "") {
|
||||
var myGuider = guiders._guiderById(nextGuiderId);
|
||||
var omitHidingOverlay = myGuider.overlay ? true : false;
|
||||
guiders.hideAll(omitHidingOverlay);
|
||||
guiders.show(nextGuiderId);
|
||||
}
|
||||
},
|
||||
|
||||
createGuider: function(passedSettings) {
|
||||
if (passedSettings === null || passedSettings === undefined) {
|
||||
passedSettings = {};
|
||||
}
|
||||
|
||||
// Extend those settings with passedSettings
|
||||
myGuider = $.extend({}, guiders._defaultSettings, passedSettings);
|
||||
myGuider.id = myGuider.id || String(Math.floor(Math.random() * 1000));
|
||||
|
||||
var guiderElement = $(guiders._htmlSkeleton);
|
||||
myGuider.elem = guiderElement;
|
||||
myGuider.elem.css("width", myGuider.width + "px");
|
||||
guiderElement.find("h1.guider_title").html(myGuider.title);
|
||||
guiderElement.find("p.guider_description").html(myGuider.description);
|
||||
|
||||
guiders._addButtons(myGuider);
|
||||
|
||||
guiderElement.hide();
|
||||
guiderElement.appendTo("body");
|
||||
guiderElement.attr("id", myGuider.id);
|
||||
|
||||
// Ensure myGuider.attachTo is a jQuery element.
|
||||
if (typeof myGuider.attachTo !== "undefined" && myGuider !== null) {
|
||||
guiders._attach(myGuider);
|
||||
guiders._styleArrow(myGuider);
|
||||
}
|
||||
|
||||
guiders._initializeOverlay();
|
||||
|
||||
guiders._guiders[myGuider.id] = myGuider;
|
||||
guiders._lastCreatedGuiderID = myGuider.id;
|
||||
|
||||
/**
|
||||
* If the URL of the current window is of the form
|
||||
* http://www.myurl.com/mypage.html#guider=id
|
||||
* then show this guider.
|
||||
*/
|
||||
if (myGuider.isHashable) {
|
||||
guiders._showIfHashed(myGuider);
|
||||
}
|
||||
|
||||
return guiders;
|
||||
},
|
||||
|
||||
hideAll: function(omitHidingOverlay) {
|
||||
$(".guider").fadeOut("fast");
|
||||
if (typeof omitHidingOverlay !== "undefined" && omitHidingOverlay === true) {
|
||||
// do nothing for now
|
||||
} else {
|
||||
guiders._hideOverlay();
|
||||
}
|
||||
return guiders;
|
||||
},
|
||||
|
||||
show: function(id) {
|
||||
if (!id && guiders._lastCreatedGuiderID) {
|
||||
id = guiders._lastCreatedGuiderID;
|
||||
}
|
||||
|
||||
var myGuider = guiders._guiderById(id);
|
||||
if (myGuider.overlay) {
|
||||
guiders._showOverlay();
|
||||
}
|
||||
|
||||
guiders._attach(myGuider);
|
||||
|
||||
// You can use an onShow function to take some action before the guider is shown.
|
||||
if (myGuider.onShow) {
|
||||
myGuider.onShow(myGuider);
|
||||
}
|
||||
|
||||
myGuider.elem.fadeIn("fast");
|
||||
|
||||
var windowHeight = $(window).height();
|
||||
var scrollHeight = $(window).scrollTop();
|
||||
var guiderOffset = myGuider.elem.offset();
|
||||
var guiderElemHeight = myGuider.elem.height();
|
||||
|
||||
if (guiderOffset.top - scrollHeight < 0 ||
|
||||
guiderOffset.top + guiderElemHeight + 40 > scrollHeight + windowHeight) {
|
||||
window.scrollTo(0, Math.max(guiderOffset.top + (guiderElemHeight / 2) - (windowHeight / 2), 0));
|
||||
}
|
||||
|
||||
guiders._currentGuiderID = id;
|
||||
return guiders;
|
||||
}
|
||||
};
|
||||
|
||||
return guiders;
|
||||
}).call(this, jQuery);
|
43
public/javascripts/admin/tutorial.js
Normal file
43
public/javascripts/admin/tutorial.js
Normal file
@ -0,0 +1,43 @@
|
||||
$(document).ready(function(){
|
||||
|
||||
guiders.createGuider({
|
||||
description: "Welcome to LocomotiveCMS, <br /><br /> This guide will help you get up and running with locomotive",
|
||||
buttons: [{name: "Quit Guide", onclick: guiders.hideAll},
|
||||
{name: "Continue", onclick: guiders.next}],
|
||||
id: "welcome",
|
||||
next: "help",
|
||||
overlay: true,
|
||||
title: "Welcome to Locomotive CMS"
|
||||
});
|
||||
|
||||
guiders.createGuider({
|
||||
attachTo: "#tutorial",
|
||||
buttons: [{name: "Close", onclick: guiders.hideAll}],
|
||||
description: "How To:<ul>\
|
||||
<li><a id='new_page_tutorial' href=\"/admin/pages#guider=newpage\">Make a new Page</a></li>\
|
||||
<li><a href=\"\">Edit a Page</a></li></ul>",
|
||||
id: "help",
|
||||
position: 6,
|
||||
width: 200,
|
||||
title: "Locomotive Tutorials",
|
||||
});
|
||||
|
||||
$('#new_page_tutorial').click(function(){
|
||||
guiders.hideAll();
|
||||
guiders.show('newpage');
|
||||
return false;
|
||||
});
|
||||
|
||||
|
||||
guiders.createGuider({
|
||||
attachTo: "#newpage",
|
||||
buttons: [{name: "Close", onclick: guiders.hideAll}],
|
||||
description: "Click above to make a new page",
|
||||
id: "newpage",
|
||||
position: 6,
|
||||
width: 200,
|
||||
title: "Make a new page",
|
||||
});
|
||||
|
||||
|
||||
});
|
118
public/stylesheets/admin/guiders-1.1.0.css
Executable file
118
public/stylesheets/admin/guiders-1.1.0.css
Executable file
@ -0,0 +1,118 @@
|
||||
.guider {
|
||||
background: #FFF;
|
||||
border: 1px solid #666;
|
||||
font-family: arial;
|
||||
position: absolute;
|
||||
outline: none;
|
||||
z-index: 100000005 !important;
|
||||
padding: 4px 12px;
|
||||
width: 500px;
|
||||
z-index: 100;
|
||||
|
||||
/* Shadow */
|
||||
-moz-box-shadow: 0 0px 8px #111;
|
||||
-webkit-box-shadow: 0 0px 8px #111;
|
||||
box-shadow: 0 0px 8px #111;
|
||||
/* End shadow */
|
||||
|
||||
/* Rounded corners */
|
||||
-moz-border-radius: 4px;
|
||||
-webkit-border-radius: 4px;
|
||||
border-radius: 4px;
|
||||
/* End rounded corners */
|
||||
}
|
||||
|
||||
.guider_buttons {
|
||||
height: 36px;
|
||||
position: relative;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.guider_content {
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.guider_content h1 {
|
||||
color: #1054AA;
|
||||
font-size: 21px;
|
||||
}
|
||||
|
||||
.guider_content p {
|
||||
color: #333;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.guider_button {
|
||||
background: -moz-linear-gradient(top, #5CA9FF 0%, #3D79C3 100%);
|
||||
background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #5CA9FF), color-stop(100%, #3D79C3));
|
||||
background-color: #4A95E0; /* overruled by background gradient, in browsers where they exist */
|
||||
border: solid 1px #4B5D7E;
|
||||
color: #FFF;
|
||||
cursor: pointer;
|
||||
display: inline-block;
|
||||
float: right;
|
||||
font-size: 75%;
|
||||
font-weight: bold;
|
||||
margin-left: 6px;
|
||||
min-width: 40px;
|
||||
padding: 3px 5px;
|
||||
text-align: center;
|
||||
text-decoration: none;
|
||||
/* Rounded corners */
|
||||
-moz-border-radius: 2px;
|
||||
-webkit-border-radius: 2px;
|
||||
border-radius: 2px;
|
||||
/* End rounded corners */
|
||||
}
|
||||
|
||||
#guider_overlay {
|
||||
background-color: #000;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
position: fixed;
|
||||
top: 0px;
|
||||
left: 0px;
|
||||
opacity: 0.5;
|
||||
-ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=50)";
|
||||
filter: alpha(opacity=50);
|
||||
z-index: 10;
|
||||
}
|
||||
|
||||
.guider_arrow {
|
||||
width: 42px;
|
||||
height: 42px;
|
||||
position: absolute;
|
||||
display: none;
|
||||
background-repeat: no-repeat;
|
||||
z-index: 100000006 !important;
|
||||
|
||||
/**
|
||||
* For optimization, the arrows image is inlined in the css below.
|
||||
*
|
||||
* To use your own arrows image, replace this background-image with your own arrows.
|
||||
* It should have four arrows, top, right, left, and down.
|
||||
*/
|
||||
background-image: url();
|
||||
*background-image: url('guider_arrows.png');
|
||||
}
|
||||
|
||||
.guider_arrow_right {
|
||||
display: block;
|
||||
background-position: 0px 0px;
|
||||
right: -42px;
|
||||
}
|
||||
.guider_arrow_down {
|
||||
display: block;
|
||||
background-position: 0px -42px;
|
||||
bottom: -42px;
|
||||
}
|
||||
.guider_arrow_up {
|
||||
display: block;
|
||||
background-position: 0px -126px;
|
||||
top: -42px;
|
||||
}
|
||||
.guider_arrow_left {
|
||||
display: block;
|
||||
background-position: 0px -84px;
|
||||
left: -42px;
|
||||
}
|
Loading…
Reference in New Issue
Block a user