1245 lines
No EOL
44 KiB
JavaScript
1245 lines
No EOL
44 KiB
JavaScript
|
|
|
|
/* ---- plugins/Sidebar/media/Class.coffee ---- */
|
|
|
|
|
|
(function() {
|
|
var Class,
|
|
slice = [].slice;
|
|
|
|
Class = (function() {
|
|
function Class() {}
|
|
|
|
Class.prototype.trace = true;
|
|
|
|
Class.prototype.log = function() {
|
|
var args;
|
|
args = 1 <= arguments.length ? slice.call(arguments, 0) : [];
|
|
if (!this.trace) {
|
|
return;
|
|
}
|
|
if (typeof console === 'undefined') {
|
|
return;
|
|
}
|
|
args.unshift("[" + this.constructor.name + "]");
|
|
console.log.apply(console, args);
|
|
return this;
|
|
};
|
|
|
|
Class.prototype.logStart = function() {
|
|
var args, name;
|
|
name = arguments[0], args = 2 <= arguments.length ? slice.call(arguments, 1) : [];
|
|
if (!this.trace) {
|
|
return;
|
|
}
|
|
this.logtimers || (this.logtimers = {});
|
|
this.logtimers[name] = +(new Date);
|
|
if (args.length > 0) {
|
|
this.log.apply(this, ["" + name].concat(slice.call(args), ["(started)"]));
|
|
}
|
|
return this;
|
|
};
|
|
|
|
Class.prototype.logEnd = function() {
|
|
var args, ms, name;
|
|
name = arguments[0], args = 2 <= arguments.length ? slice.call(arguments, 1) : [];
|
|
ms = +(new Date) - this.logtimers[name];
|
|
this.log.apply(this, ["" + name].concat(slice.call(args), ["(Done in " + ms + "ms)"]));
|
|
return this;
|
|
};
|
|
|
|
return Class;
|
|
|
|
})();
|
|
|
|
window.Class = Class;
|
|
|
|
}).call(this);
|
|
|
|
|
|
/* ---- plugins/Sidebar/media/Menu.coffee ---- */
|
|
|
|
|
|
(function() {
|
|
var Menu,
|
|
slice = [].slice;
|
|
|
|
Menu = (function() {
|
|
function Menu(button) {
|
|
this.button = button;
|
|
this.elem = $(".menu.template").clone().removeClass("template");
|
|
this.elem.appendTo("body");
|
|
this.items = [];
|
|
}
|
|
|
|
Menu.prototype.show = function() {
|
|
var button_pos, left;
|
|
if (window.visible_menu && window.visible_menu.button[0] === this.button[0]) {
|
|
window.visible_menu.hide();
|
|
return this.hide();
|
|
} else {
|
|
button_pos = this.button.offset();
|
|
left = button_pos.left;
|
|
this.elem.css({
|
|
"top": button_pos.top + this.button.outerHeight(),
|
|
"left": left
|
|
});
|
|
this.button.addClass("menu-active");
|
|
this.elem.addClass("visible");
|
|
if (this.elem.position().left + this.elem.width() + 20 > window.innerWidth) {
|
|
this.elem.css("left", window.innerWidth - this.elem.width() - 20);
|
|
}
|
|
if (window.visible_menu) {
|
|
window.visible_menu.hide();
|
|
}
|
|
return window.visible_menu = this;
|
|
}
|
|
};
|
|
|
|
Menu.prototype.hide = function() {
|
|
this.elem.removeClass("visible");
|
|
this.button.removeClass("menu-active");
|
|
return window.visible_menu = null;
|
|
};
|
|
|
|
Menu.prototype.addItem = function(title, cb) {
|
|
var item;
|
|
item = $(".menu-item.template", this.elem).clone().removeClass("template");
|
|
item.html(title);
|
|
item.on("click", (function(_this) {
|
|
return function() {
|
|
if (!cb(item)) {
|
|
_this.hide();
|
|
}
|
|
return false;
|
|
};
|
|
})(this));
|
|
item.appendTo(this.elem);
|
|
this.items.push(item);
|
|
return item;
|
|
};
|
|
|
|
Menu.prototype.log = function() {
|
|
var args;
|
|
args = 1 <= arguments.length ? slice.call(arguments, 0) : [];
|
|
return console.log.apply(console, ["[Menu]"].concat(slice.call(args)));
|
|
};
|
|
|
|
return Menu;
|
|
|
|
})();
|
|
|
|
window.Menu = Menu;
|
|
|
|
$("body").on("click", function(e) {
|
|
if (window.visible_menu && e.target !== window.visible_menu.button[0] && $(e.target).parent()[0] !== window.visible_menu.elem[0]) {
|
|
return window.visible_menu.hide();
|
|
}
|
|
});
|
|
|
|
}).call(this);
|
|
|
|
|
|
/* ---- plugins/Sidebar/media/RateLimit.coffee ---- */
|
|
|
|
|
|
(function() {
|
|
var call_after_interval, limits;
|
|
|
|
limits = {};
|
|
|
|
call_after_interval = {};
|
|
|
|
window.RateLimit = function(interval, fn) {
|
|
if (!limits[fn]) {
|
|
call_after_interval[fn] = false;
|
|
fn();
|
|
return limits[fn] = setTimeout((function() {
|
|
if (call_after_interval[fn]) {
|
|
fn();
|
|
}
|
|
delete limits[fn];
|
|
return delete call_after_interval[fn];
|
|
}), interval);
|
|
} else {
|
|
return call_after_interval[fn] = true;
|
|
}
|
|
};
|
|
|
|
}).call(this);
|
|
|
|
|
|
/* ---- plugins/Sidebar/media/Scrollable.js ---- */
|
|
|
|
|
|
/* via http://jsfiddle.net/elGrecode/00dgurnn/ */
|
|
|
|
window.initScrollable = function () {
|
|
|
|
var scrollContainer = document.querySelector('.scrollable'),
|
|
scrollContentWrapper = document.querySelector('.scrollable .content-wrapper'),
|
|
scrollContent = document.querySelector('.scrollable .content'),
|
|
contentPosition = 0,
|
|
scrollerBeingDragged = false,
|
|
scroller,
|
|
topPosition,
|
|
scrollerHeight;
|
|
|
|
function calculateScrollerHeight() {
|
|
// *Calculation of how tall scroller should be
|
|
var visibleRatio = scrollContainer.offsetHeight / scrollContentWrapper.scrollHeight;
|
|
if (visibleRatio == 1)
|
|
scroller.style.display = "none";
|
|
else
|
|
scroller.style.display = "block";
|
|
return visibleRatio * scrollContainer.offsetHeight;
|
|
}
|
|
|
|
function moveScroller(evt) {
|
|
// Move Scroll bar to top offset
|
|
var scrollPercentage = evt.target.scrollTop / scrollContentWrapper.scrollHeight;
|
|
topPosition = scrollPercentage * (scrollContainer.offsetHeight - 5); // 5px arbitrary offset so scroll bar doesn't move too far beyond content wrapper bounding box
|
|
scroller.style.top = topPosition + 'px';
|
|
}
|
|
|
|
function startDrag(evt) {
|
|
normalizedPosition = evt.pageY;
|
|
contentPosition = scrollContentWrapper.scrollTop;
|
|
scrollerBeingDragged = true;
|
|
window.addEventListener('mousemove', scrollBarScroll);
|
|
return false;
|
|
}
|
|
|
|
function stopDrag(evt) {
|
|
scrollerBeingDragged = false;
|
|
window.removeEventListener('mousemove', scrollBarScroll);
|
|
}
|
|
|
|
function scrollBarScroll(evt) {
|
|
if (scrollerBeingDragged === true) {
|
|
evt.preventDefault();
|
|
var mouseDifferential = evt.pageY - normalizedPosition;
|
|
var scrollEquivalent = mouseDifferential * (scrollContentWrapper.scrollHeight / scrollContainer.offsetHeight);
|
|
scrollContentWrapper.scrollTop = contentPosition + scrollEquivalent;
|
|
}
|
|
}
|
|
|
|
function updateHeight() {
|
|
scrollerHeight = calculateScrollerHeight() - 10;
|
|
scroller.style.height = scrollerHeight + 'px';
|
|
}
|
|
|
|
function createScroller() {
|
|
// *Creates scroller element and appends to '.scrollable' div
|
|
// create scroller element
|
|
scroller = document.createElement("div");
|
|
scroller.className = 'scroller';
|
|
|
|
// determine how big scroller should be based on content
|
|
scrollerHeight = calculateScrollerHeight() - 10;
|
|
|
|
if (scrollerHeight / scrollContainer.offsetHeight < 1) {
|
|
// *If there is a need to have scroll bar based on content size
|
|
scroller.style.height = scrollerHeight + 'px';
|
|
|
|
// append scroller to scrollContainer div
|
|
scrollContainer.appendChild(scroller);
|
|
|
|
// show scroll path divot
|
|
scrollContainer.className += ' showScroll';
|
|
|
|
// attach related draggable listeners
|
|
scroller.addEventListener('mousedown', startDrag);
|
|
window.addEventListener('mouseup', stopDrag);
|
|
}
|
|
|
|
}
|
|
|
|
createScroller();
|
|
|
|
|
|
// *** Listeners ***
|
|
scrollContentWrapper.addEventListener('scroll', moveScroller);
|
|
|
|
return updateHeight;
|
|
};
|
|
|
|
|
|
/* ---- plugins/Sidebar/media/Sidebar.coffee ---- */
|
|
|
|
|
|
(function() {
|
|
var Sidebar, wrapper,
|
|
bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; },
|
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
|
hasProp = {}.hasOwnProperty,
|
|
indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; };
|
|
|
|
Sidebar = (function(superClass) {
|
|
extend(Sidebar, superClass);
|
|
|
|
function Sidebar(wrapper1) {
|
|
this.wrapper = wrapper1;
|
|
this.unloadGlobe = bind(this.unloadGlobe, this);
|
|
this.displayGlobe = bind(this.displayGlobe, this);
|
|
this.loadGlobe = bind(this.loadGlobe, this);
|
|
this.animDrag = bind(this.animDrag, this);
|
|
this.setHtmlTag = bind(this.setHtmlTag, this);
|
|
this.waitMove = bind(this.waitMove, this);
|
|
this.resized = bind(this.resized, this);
|
|
this.tag = null;
|
|
this.container = null;
|
|
this.opened = false;
|
|
this.width = 410;
|
|
this.fixbutton = $(".fixbutton");
|
|
this.fixbutton_addx = 0;
|
|
this.fixbutton_initx = 0;
|
|
this.fixbutton_targetx = 0;
|
|
this.page_width = $(window).width();
|
|
this.frame = $("#inner-iframe");
|
|
this.initFixbutton();
|
|
this.dragStarted = 0;
|
|
this.globe = null;
|
|
this.preload_html = null;
|
|
this.original_set_site_info = this.wrapper.setSiteInfo;
|
|
if (false) {
|
|
this.startDrag();
|
|
this.moved();
|
|
this.fixbutton_targetx = this.fixbutton_initx - this.width;
|
|
this.stopDrag();
|
|
}
|
|
}
|
|
|
|
Sidebar.prototype.initFixbutton = function() {
|
|
this.fixbutton.on("mousedown touchstart", (function(_this) {
|
|
return function(e) {
|
|
if (e.button > 0) {
|
|
return;
|
|
}
|
|
e.preventDefault();
|
|
_this.fixbutton.off("click touchend touchcancel");
|
|
_this.fixbutton.off("mousemove touchmove");
|
|
_this.dragStarted = +(new Date);
|
|
return _this.fixbutton.one("mousemove touchmove", function(e) {
|
|
var mousex;
|
|
mousex = e.pageX;
|
|
if (!mousex) {
|
|
mousex = e.originalEvent.touches[0].pageX;
|
|
}
|
|
_this.fixbutton_addx = _this.fixbutton.offset().left - mousex;
|
|
return _this.startDrag();
|
|
});
|
|
};
|
|
})(this));
|
|
this.fixbutton.parent().on("click touchend touchcancel", (function(_this) {
|
|
return function(e) {
|
|
if ((+(new Date)) - _this.dragStarted < 100) {
|
|
window.top.location = _this.fixbutton.find(".fixbutton-bg").attr("href");
|
|
}
|
|
return _this.stopDrag();
|
|
};
|
|
})(this));
|
|
this.resized();
|
|
return $(window).on("resize", this.resized);
|
|
};
|
|
|
|
Sidebar.prototype.resized = function() {
|
|
this.page_width = $(window).width();
|
|
this.fixbutton_initx = this.page_width - 75;
|
|
if (this.opened) {
|
|
return this.fixbutton.css({
|
|
left: this.fixbutton_initx - this.width
|
|
});
|
|
} else {
|
|
return this.fixbutton.css({
|
|
left: this.fixbutton_initx
|
|
});
|
|
}
|
|
};
|
|
|
|
Sidebar.prototype.startDrag = function() {
|
|
this.log("startDrag");
|
|
this.fixbutton_targetx = this.fixbutton_initx;
|
|
this.fixbutton.addClass("dragging");
|
|
$("<div class='drag-bg'></div>").appendTo(document.body);
|
|
if (navigator.userAgent.indexOf('MSIE') !== -1 || navigator.appVersion.indexOf('Trident/') > 0) {
|
|
this.fixbutton.css("pointer-events", "none");
|
|
}
|
|
this.fixbutton.one("click", (function(_this) {
|
|
return function(e) {
|
|
_this.stopDrag();
|
|
_this.fixbutton.removeClass("dragging");
|
|
if (Math.abs(_this.fixbutton.offset().left - _this.fixbutton_initx) > 5) {
|
|
return e.preventDefault();
|
|
}
|
|
};
|
|
})(this));
|
|
this.fixbutton.parents().on("mousemove touchmove", this.animDrag);
|
|
this.fixbutton.parents().on("mousemove touchmove", this.waitMove);
|
|
return this.fixbutton.parents().on("mouseup touchend touchend touchcancel", (function(_this) {
|
|
return function(e) {
|
|
e.preventDefault();
|
|
return _this.stopDrag();
|
|
};
|
|
})(this));
|
|
};
|
|
|
|
Sidebar.prototype.waitMove = function(e) {
|
|
if (Math.abs(this.fixbutton.offset().left - this.fixbutton_targetx) > 10 && (+(new Date)) - this.dragStarted > 100) {
|
|
this.moved();
|
|
return this.fixbutton.parents().off("mousemove touchmove", this.waitMove);
|
|
}
|
|
};
|
|
|
|
Sidebar.prototype.moved = function() {
|
|
var img;
|
|
this.log("Moved");
|
|
this.createHtmltag();
|
|
$(document.body).css("perspective", "1000px").addClass("body-sidebar");
|
|
$(window).off("resize");
|
|
$(window).on("resize", (function(_this) {
|
|
return function() {
|
|
$(document.body).css("height", $(window).height());
|
|
_this.scrollable();
|
|
return _this.resized();
|
|
};
|
|
})(this));
|
|
$(window).trigger("resize");
|
|
this.wrapper.setSiteInfo = (function(_this) {
|
|
return function(site_info) {
|
|
_this.setSiteInfo(site_info);
|
|
return _this.original_set_site_info.apply(_this.wrapper, arguments);
|
|
};
|
|
})(this);
|
|
img = new Image();
|
|
return img.src = "/uimedia/globe/world.jpg";
|
|
};
|
|
|
|
Sidebar.prototype.setSiteInfo = function(site_info) {
|
|
RateLimit(1500, (function(_this) {
|
|
return function() {
|
|
return _this.updateHtmlTag();
|
|
};
|
|
})(this));
|
|
return RateLimit(30000, (function(_this) {
|
|
return function() {
|
|
return _this.displayGlobe();
|
|
};
|
|
})(this));
|
|
};
|
|
|
|
Sidebar.prototype.createHtmltag = function() {
|
|
this.when_loaded = $.Deferred();
|
|
if (!this.container) {
|
|
this.container = $("<div class=\"sidebar-container\"><div class=\"sidebar scrollable\"><div class=\"content-wrapper\"><div class=\"content\">\n</div></div></div></div>");
|
|
this.container.appendTo(document.body);
|
|
this.tag = this.container.find(".sidebar");
|
|
this.updateHtmlTag();
|
|
return this.scrollable = window.initScrollable();
|
|
}
|
|
};
|
|
|
|
Sidebar.prototype.updateHtmlTag = function() {
|
|
if (this.preload_html) {
|
|
this.setHtmlTag(this.preload_html);
|
|
return this.preload_html = null;
|
|
} else {
|
|
return this.wrapper.ws.cmd("sidebarGetHtmlTag", {}, this.setHtmlTag);
|
|
}
|
|
};
|
|
|
|
Sidebar.prototype.setHtmlTag = function(res) {
|
|
if (this.tag.find(".content").children().length === 0) {
|
|
this.log("Creating content");
|
|
this.container.addClass("loaded");
|
|
morphdom(this.tag.find(".content")[0], '<div class="content">' + res + '</div>');
|
|
return this.when_loaded.resolve();
|
|
} else {
|
|
this.log("Patching content");
|
|
return morphdom(this.tag.find(".content")[0], '<div class="content">' + res + '</div>', {
|
|
onBeforeMorphEl: function(from_el, to_el) {
|
|
if (from_el.className === "globe" || from_el.className.indexOf("noupdate") >= 0) {
|
|
return false;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
Sidebar.prototype.animDrag = function(e) {
|
|
var mousex, overdrag, overdrag_percent, targetx;
|
|
mousex = e.pageX;
|
|
if (!mousex) {
|
|
mousex = e.originalEvent.touches[0].pageX;
|
|
}
|
|
overdrag = this.fixbutton_initx - this.width - mousex;
|
|
if (overdrag > 0) {
|
|
overdrag_percent = 1 + overdrag / 300;
|
|
mousex = (mousex + (this.fixbutton_initx - this.width) * overdrag_percent) / (1 + overdrag_percent);
|
|
}
|
|
targetx = this.fixbutton_initx - mousex - this.fixbutton_addx;
|
|
this.fixbutton[0].style.left = (mousex + this.fixbutton_addx) + "px";
|
|
if (this.tag) {
|
|
this.tag[0].style.transform = "translateX(" + (0 - targetx) + "px)";
|
|
}
|
|
if ((!this.opened && targetx > this.width / 3) || (this.opened && targetx > this.width * 0.9)) {
|
|
return this.fixbutton_targetx = this.fixbutton_initx - this.width;
|
|
} else {
|
|
return this.fixbutton_targetx = this.fixbutton_initx;
|
|
}
|
|
};
|
|
|
|
Sidebar.prototype.stopDrag = function() {
|
|
var targetx;
|
|
this.fixbutton.parents().off("mousemove touchmove");
|
|
this.fixbutton.off("mousemove touchmove");
|
|
this.fixbutton.css("pointer-events", "");
|
|
$(".drag-bg").remove();
|
|
if (!this.fixbutton.hasClass("dragging")) {
|
|
return;
|
|
}
|
|
this.fixbutton.removeClass("dragging");
|
|
if (this.fixbutton_targetx !== this.fixbutton.offset().left) {
|
|
this.fixbutton.stop().animate({
|
|
"left": this.fixbutton_targetx
|
|
}, 500, "easeOutBack", (function(_this) {
|
|
return function() {
|
|
if (_this.fixbutton_targetx === _this.fixbutton_initx) {
|
|
_this.fixbutton.css("left", "auto");
|
|
} else {
|
|
_this.fixbutton.css("left", _this.fixbutton_targetx);
|
|
}
|
|
return $(".fixbutton-bg").trigger("mouseout");
|
|
};
|
|
})(this));
|
|
if (this.fixbutton_targetx === this.fixbutton_initx) {
|
|
targetx = 0;
|
|
this.opened = false;
|
|
} else {
|
|
targetx = this.width;
|
|
if (this.opened) {
|
|
onOpened();
|
|
} else {
|
|
this.when_loaded.done((function(_this) {
|
|
return function() {
|
|
return _this.onOpened();
|
|
};
|
|
})(this));
|
|
}
|
|
this.opened = true;
|
|
}
|
|
if (this.tag) {
|
|
this.tag.css("transition", "0.4s ease-out");
|
|
this.tag.css("transform", "translateX(-" + targetx + "px)").one(transitionEnd, (function(_this) {
|
|
return function() {
|
|
_this.tag.css("transition", "");
|
|
if (!_this.opened) {
|
|
_this.container.remove();
|
|
_this.container = null;
|
|
_this.tag.remove();
|
|
return _this.tag = null;
|
|
}
|
|
};
|
|
})(this));
|
|
}
|
|
this.log("stopdrag", "opened:", this.opened);
|
|
if (!this.opened) {
|
|
return this.onClosed();
|
|
}
|
|
}
|
|
};
|
|
|
|
Sidebar.prototype.onOpened = function() {
|
|
var menu;
|
|
this.log("Opened");
|
|
this.scrollable();
|
|
this.tag.find("#checkbox-owned").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
return setTimeout((function() {
|
|
return _this.scrollable();
|
|
}), 300);
|
|
};
|
|
})(this));
|
|
this.tag.find("#button-sitelimit").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
_this.wrapper.ws.cmd("siteSetLimit", $("#input-sitelimit").val(), function(res) {
|
|
if (res === "ok") {
|
|
_this.wrapper.notifications.add("done-sitelimit", "done", "Site storage limit modified!", 5000);
|
|
}
|
|
return _this.updateHtmlTag();
|
|
});
|
|
return false;
|
|
};
|
|
})(this));
|
|
this.tag.find("#button-dbreload").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
_this.wrapper.ws.cmd("dbReload", [], function() {
|
|
_this.wrapper.notifications.add("done-dbreload", "done", "Database schema reloaded!", 5000);
|
|
return _this.updateHtmlTag();
|
|
});
|
|
return false;
|
|
};
|
|
})(this));
|
|
this.tag.find("#button-dbrebuild").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
_this.wrapper.notifications.add("done-dbrebuild", "info", "Database rebuilding....");
|
|
_this.wrapper.ws.cmd("dbRebuild", [], function() {
|
|
_this.wrapper.notifications.add("done-dbrebuild", "done", "Database rebuilt!", 5000);
|
|
return _this.updateHtmlTag();
|
|
});
|
|
return false;
|
|
};
|
|
})(this));
|
|
this.tag.find("#button-update").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
_this.tag.find("#button-update").addClass("loading");
|
|
_this.wrapper.ws.cmd("siteUpdate", _this.wrapper.site_info.address, function() {
|
|
_this.wrapper.notifications.add("done-updated", "done", "Site updated!", 5000);
|
|
return _this.tag.find("#button-update").removeClass("loading");
|
|
});
|
|
return false;
|
|
};
|
|
})(this));
|
|
this.tag.find("#button-pause").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
_this.tag.find("#button-pause").addClass("hidden");
|
|
_this.wrapper.ws.cmd("sitePause", _this.wrapper.site_info.address);
|
|
return false;
|
|
};
|
|
})(this));
|
|
this.tag.find("#button-resume").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
_this.tag.find("#button-resume").addClass("hidden");
|
|
_this.wrapper.ws.cmd("siteResume", _this.wrapper.site_info.address);
|
|
return false;
|
|
};
|
|
})(this));
|
|
this.tag.find("#button-delete").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
_this.wrapper.displayConfirm("Are you sure?", ["Delete this site", "Blacklist"], function(confirmed) {
|
|
if (confirmed === 1) {
|
|
_this.tag.find("#button-delete").addClass("loading");
|
|
return _this.wrapper.ws.cmd("siteDelete", _this.wrapper.site_info.address, function() {
|
|
return document.location = $(".fixbutton-bg").attr("href");
|
|
});
|
|
} else if (confirmed === 2) {
|
|
return _this.wrapper.displayPrompt("Blacklist this site", "text", "Delete and Blacklist", "Reason", function(reason) {
|
|
_this.tag.find("#button-delete").addClass("loading");
|
|
_this.wrapper.ws.cmd("blacklistAdd", [_this.wrapper.site_info.address, reason]);
|
|
return _this.wrapper.ws.cmd("siteDelete", _this.wrapper.site_info.address, function() {
|
|
return document.location = $(".fixbutton-bg").attr("href");
|
|
});
|
|
});
|
|
}
|
|
});
|
|
return false;
|
|
};
|
|
})(this));
|
|
this.tag.find("#checkbox-owned").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
return _this.wrapper.ws.cmd("siteSetOwned", [_this.tag.find("#checkbox-owned").is(":checked")]);
|
|
};
|
|
})(this));
|
|
this.tag.find("#checkbox-autodownloadoptional").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
return _this.wrapper.ws.cmd("siteSetAutodownloadoptional", [_this.tag.find("#checkbox-autodownloadoptional").is(":checked")]);
|
|
};
|
|
})(this));
|
|
this.tag.find("#button-identity").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
_this.wrapper.ws.cmd("certSelect");
|
|
return false;
|
|
};
|
|
})(this));
|
|
this.tag.find("#checkbox-owned").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
return _this.wrapper.ws.cmd("siteSetOwned", [_this.tag.find("#checkbox-owned").is(":checked")]);
|
|
};
|
|
})(this));
|
|
this.tag.find("#button-settings").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
_this.wrapper.ws.cmd("fileGet", "content.json", function(res) {
|
|
var data, json_raw;
|
|
data = JSON.parse(res);
|
|
data["title"] = $("#settings-title").val();
|
|
data["description"] = $("#settings-description").val();
|
|
json_raw = unescape(encodeURIComponent(JSON.stringify(data, void 0, '\t')));
|
|
return _this.wrapper.ws.cmd("fileWrite", ["content.json", btoa(json_raw), true], function(res) {
|
|
if (res !== "ok") {
|
|
return _this.wrapper.notifications.add("file-write", "error", "File write error: " + res);
|
|
} else {
|
|
_this.wrapper.notifications.add("file-write", "done", "Site settings saved!", 5000);
|
|
if (_this.wrapper.site_info.privatekey) {
|
|
_this.wrapper.ws.cmd("siteSign", {
|
|
privatekey: "stored",
|
|
inner_path: "content.json",
|
|
update_changed_files: true
|
|
});
|
|
}
|
|
return _this.updateHtmlTag();
|
|
}
|
|
});
|
|
});
|
|
return false;
|
|
};
|
|
})(this));
|
|
$(document).on("click touchend", (function(_this) {
|
|
return function() {
|
|
_this.tag.find("#button-sign-publish-menu").removeClass("visible");
|
|
return _this.tag.find(".contents + .flex").removeClass("sign-publish-flex");
|
|
};
|
|
})(this));
|
|
menu = new Menu(this.tag.find("#menu-sign-publish"));
|
|
menu.elem.css("margin-top", "-130px");
|
|
menu.addItem("Sign", (function(_this) {
|
|
return function() {
|
|
var inner_path;
|
|
inner_path = _this.tag.find("#input-contents").val();
|
|
_this.wrapper.ws.cmd("fileRules", {
|
|
inner_path: inner_path
|
|
}, function(res) {
|
|
var ref;
|
|
if (_this.wrapper.site_info.privatekey || (ref = _this.wrapper.site_info.auth_address, indexOf.call(res.signers, ref) >= 0)) {
|
|
return _this.wrapper.ws.cmd("siteSign", {
|
|
privatekey: "stored",
|
|
inner_path: inner_path,
|
|
update_changed_files: true
|
|
}, function(res) {
|
|
if (res === "ok") {
|
|
return _this.wrapper.notifications.add("sign", "done", inner_path + " Signed!", 5000);
|
|
}
|
|
});
|
|
} else {
|
|
return _this.wrapper.displayPrompt("Enter your private key:", "password", "Sign", "", function(privatekey) {
|
|
return _this.wrapper.ws.cmd("siteSign", {
|
|
privatekey: privatekey,
|
|
inner_path: inner_path,
|
|
update_changed_files: true
|
|
}, function(res) {
|
|
if (res === "ok") {
|
|
return _this.wrapper.notifications.add("sign", "done", inner_path + " Signed!", 5000);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
});
|
|
_this.tag.find(".contents + .flex").removeClass("active");
|
|
return menu.hide();
|
|
};
|
|
})(this));
|
|
menu.addItem("Publish", (function(_this) {
|
|
return function() {
|
|
var inner_path;
|
|
inner_path = _this.tag.find("#input-contents").val();
|
|
_this.wrapper.ws.cmd("sitePublish", {
|
|
"inner_path": inner_path,
|
|
"sign": false
|
|
});
|
|
_this.tag.find(".contents + .flex").removeClass("active");
|
|
return menu.hide();
|
|
};
|
|
})(this));
|
|
this.tag.find("#menu-sign-publish").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
if (window.visible_menu === menu) {
|
|
_this.tag.find(".contents + .flex").removeClass("active");
|
|
menu.hide();
|
|
} else {
|
|
_this.tag.find(".contents + .flex").addClass("active");
|
|
_this.tag.find(".content-wrapper").prop("scrollTop", 10000);
|
|
menu.show();
|
|
}
|
|
return false;
|
|
};
|
|
})(this));
|
|
$("body").on("click", (function(_this) {
|
|
return function() {
|
|
if (_this.tag) {
|
|
return _this.tag.find(".contents + .flex").removeClass("active");
|
|
}
|
|
};
|
|
})(this));
|
|
this.tag.find("#button-sign-publish").off("click touchend").on("click touchend", (function(_this) {
|
|
return function() {
|
|
var inner_path;
|
|
inner_path = _this.tag.find("#input-contents").val();
|
|
_this.wrapper.ws.cmd("fileRules", {
|
|
inner_path: inner_path
|
|
}, function(res) {
|
|
var ref;
|
|
if (_this.wrapper.site_info.privatekey || (ref = _this.wrapper.site_info.auth_address, indexOf.call(res.signers, ref) >= 0)) {
|
|
return _this.wrapper.ws.cmd("sitePublish", {
|
|
privatekey: "stored",
|
|
inner_path: inner_path,
|
|
sign: true
|
|
}, function(res) {
|
|
if (res === "ok") {
|
|
return _this.wrapper.notifications.add("sign", "done", inner_path + " Signed and published!", 5000);
|
|
}
|
|
});
|
|
} else {
|
|
return _this.wrapper.displayPrompt("Enter your private key:", "password", "Sign", "", function(privatekey) {
|
|
return _this.wrapper.ws.cmd("sitePublish", {
|
|
privatekey: privatekey,
|
|
inner_path: inner_path,
|
|
sign: true
|
|
}, function(res) {
|
|
if (res === "ok") {
|
|
return _this.wrapper.notifications.add("sign", "done", inner_path + " Signed and published!", 5000);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
});
|
|
return false;
|
|
};
|
|
})(this));
|
|
this.tag.find(".close").off("click touchend").on("click touchend", (function(_this) {
|
|
return function(e) {
|
|
_this.startDrag();
|
|
_this.stopDrag();
|
|
return false;
|
|
};
|
|
})(this));
|
|
return this.loadGlobe();
|
|
};
|
|
|
|
Sidebar.prototype.onClosed = function() {
|
|
$(window).off("resize");
|
|
$(window).on("resize", this.resized);
|
|
$(document.body).css("transition", "0.6s ease-in-out").removeClass("body-sidebar").on(transitionEnd, (function(_this) {
|
|
return function(e) {
|
|
if (e.target === document.body) {
|
|
$(document.body).css("height", "auto").css("perspective", "").css("transition", "").off(transitionEnd);
|
|
return _this.unloadGlobe();
|
|
}
|
|
};
|
|
})(this));
|
|
return this.wrapper.setSiteInfo = this.original_set_site_info;
|
|
};
|
|
|
|
Sidebar.prototype.loadGlobe = function() {
|
|
console.log("loadGlobe", this.tag.find(".globe").hasClass("loading"));
|
|
if (this.tag.find(".globe").hasClass("loading")) {
|
|
return setTimeout(((function(_this) {
|
|
return function() {
|
|
if (typeof DAT === "undefined") {
|
|
return $.getScript("/uimedia/globe/all.js", _this.displayGlobe);
|
|
} else {
|
|
return _this.displayGlobe();
|
|
}
|
|
};
|
|
})(this)), 600);
|
|
}
|
|
};
|
|
|
|
Sidebar.prototype.displayGlobe = function() {
|
|
var img;
|
|
img = new Image();
|
|
img.src = "/uimedia/globe/world.jpg";
|
|
return img.onload = (function(_this) {
|
|
return function() {
|
|
return _this.wrapper.ws.cmd("sidebarGetPeers", [], function(globe_data) {
|
|
var e, ref, ref1;
|
|
if (_this.globe) {
|
|
_this.globe.scene.remove(_this.globe.points);
|
|
_this.globe.addData(globe_data, {
|
|
format: 'magnitude',
|
|
name: "hello",
|
|
animated: false
|
|
});
|
|
_this.globe.createPoints();
|
|
} else if (typeof DAT !== "undefined") {
|
|
try {
|
|
_this.globe = new DAT.Globe(_this.tag.find(".globe")[0], {
|
|
"imgDir": "/uimedia/globe/"
|
|
});
|
|
_this.globe.addData(globe_data, {
|
|
format: 'magnitude',
|
|
name: "hello"
|
|
});
|
|
_this.globe.createPoints();
|
|
_this.globe.animate();
|
|
} catch (error) {
|
|
e = error;
|
|
console.log("WebGL error", e);
|
|
if ((ref = _this.tag) != null) {
|
|
ref.find(".globe").addClass("error").text("WebGL not supported");
|
|
}
|
|
}
|
|
}
|
|
return (ref1 = _this.tag) != null ? ref1.find(".globe").removeClass("loading") : void 0;
|
|
});
|
|
};
|
|
})(this);
|
|
};
|
|
|
|
Sidebar.prototype.unloadGlobe = function() {
|
|
if (!this.globe) {
|
|
return false;
|
|
}
|
|
this.globe.unload();
|
|
return this.globe = null;
|
|
};
|
|
|
|
return Sidebar;
|
|
|
|
})(Class);
|
|
|
|
wrapper = window.wrapper;
|
|
|
|
setTimeout((function() {
|
|
return window.sidebar = new Sidebar(wrapper);
|
|
}), 500);
|
|
|
|
window.transitionEnd = 'transitionend webkitTransitionEnd oTransitionEnd otransitionend';
|
|
|
|
}).call(this);
|
|
|
|
|
|
|
|
/* ---- plugins/Sidebar/media/morphdom.js ---- */
|
|
|
|
|
|
(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.morphdom = f()}})(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
|
|
var specialElHandlers = {
|
|
/**
|
|
* Needed for IE. Apparently IE doesn't think
|
|
* that "selected" is an attribute when reading
|
|
* over the attributes using selectEl.attributes
|
|
*/
|
|
OPTION: function(fromEl, toEl) {
|
|
if ((fromEl.selected = toEl.selected)) {
|
|
fromEl.setAttribute('selected', '');
|
|
} else {
|
|
fromEl.removeAttribute('selected', '');
|
|
}
|
|
},
|
|
/**
|
|
* The "value" attribute is special for the <input> element
|
|
* since it sets the initial value. Changing the "value"
|
|
* attribute without changing the "value" property will have
|
|
* no effect since it is only used to the set the initial value.
|
|
* Similar for the "checked" attribute.
|
|
*/
|
|
/*INPUT: function(fromEl, toEl) {
|
|
fromEl.checked = toEl.checked;
|
|
fromEl.value = toEl.value;
|
|
|
|
if (!toEl.hasAttribute('checked')) {
|
|
fromEl.removeAttribute('checked');
|
|
}
|
|
|
|
if (!toEl.hasAttribute('value')) {
|
|
fromEl.removeAttribute('value');
|
|
}
|
|
}*/
|
|
};
|
|
|
|
function noop() {}
|
|
|
|
/**
|
|
* Loop over all of the attributes on the target node and make sure the
|
|
* original DOM node has the same attributes. If an attribute
|
|
* found on the original node is not on the new node then remove it from
|
|
* the original node
|
|
* @param {HTMLElement} fromNode
|
|
* @param {HTMLElement} toNode
|
|
*/
|
|
function morphAttrs(fromNode, toNode) {
|
|
var attrs = toNode.attributes;
|
|
var i;
|
|
var attr;
|
|
var attrName;
|
|
var attrValue;
|
|
var foundAttrs = {};
|
|
|
|
for (i=attrs.length-1; i>=0; i--) {
|
|
attr = attrs[i];
|
|
if (attr.specified !== false) {
|
|
attrName = attr.name;
|
|
attrValue = attr.value;
|
|
foundAttrs[attrName] = true;
|
|
|
|
if (fromNode.getAttribute(attrName) !== attrValue) {
|
|
fromNode.setAttribute(attrName, attrValue);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Delete any extra attributes found on the original DOM element that weren't
|
|
// found on the target element.
|
|
attrs = fromNode.attributes;
|
|
|
|
for (i=attrs.length-1; i>=0; i--) {
|
|
attr = attrs[i];
|
|
if (attr.specified !== false) {
|
|
attrName = attr.name;
|
|
if (!foundAttrs.hasOwnProperty(attrName)) {
|
|
fromNode.removeAttribute(attrName);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Copies the children of one DOM element to another DOM element
|
|
*/
|
|
function moveChildren(from, to) {
|
|
var curChild = from.firstChild;
|
|
while(curChild) {
|
|
var nextChild = curChild.nextSibling;
|
|
to.appendChild(curChild);
|
|
curChild = nextChild;
|
|
}
|
|
return to;
|
|
}
|
|
|
|
function morphdom(fromNode, toNode, options) {
|
|
if (!options) {
|
|
options = {};
|
|
}
|
|
|
|
if (typeof toNode === 'string') {
|
|
var newBodyEl = document.createElement('body');
|
|
newBodyEl.innerHTML = toNode;
|
|
toNode = newBodyEl.childNodes[0];
|
|
}
|
|
|
|
var savedEls = {}; // Used to save off DOM elements with IDs
|
|
var unmatchedEls = {};
|
|
var onNodeDiscarded = options.onNodeDiscarded || noop;
|
|
var onBeforeMorphEl = options.onBeforeMorphEl || noop;
|
|
var onBeforeMorphElChildren = options.onBeforeMorphElChildren || noop;
|
|
|
|
function removeNodeHelper(node, nestedInSavedEl) {
|
|
var id = node.id;
|
|
// If the node has an ID then save it off since we will want
|
|
// to reuse it in case the target DOM tree has a DOM element
|
|
// with the same ID
|
|
if (id) {
|
|
savedEls[id] = node;
|
|
} else if (!nestedInSavedEl) {
|
|
// If we are not nested in a saved element then we know that this node has been
|
|
// completely discarded and will not exist in the final DOM.
|
|
onNodeDiscarded(node);
|
|
}
|
|
|
|
if (node.nodeType === 1) {
|
|
var curChild = node.firstChild;
|
|
while(curChild) {
|
|
removeNodeHelper(curChild, nestedInSavedEl || id);
|
|
curChild = curChild.nextSibling;
|
|
}
|
|
}
|
|
}
|
|
|
|
function walkDiscardedChildNodes(node) {
|
|
if (node.nodeType === 1) {
|
|
var curChild = node.firstChild;
|
|
while(curChild) {
|
|
|
|
|
|
if (!curChild.id) {
|
|
// We only want to handle nodes that don't have an ID to avoid double
|
|
// walking the same saved element.
|
|
|
|
onNodeDiscarded(curChild);
|
|
|
|
// Walk recursively
|
|
walkDiscardedChildNodes(curChild);
|
|
}
|
|
|
|
curChild = curChild.nextSibling;
|
|
}
|
|
}
|
|
}
|
|
|
|
function removeNode(node, parentNode, alreadyVisited) {
|
|
parentNode.removeChild(node);
|
|
|
|
if (alreadyVisited) {
|
|
if (!node.id) {
|
|
onNodeDiscarded(node);
|
|
walkDiscardedChildNodes(node);
|
|
}
|
|
} else {
|
|
removeNodeHelper(node);
|
|
}
|
|
}
|
|
|
|
function morphEl(fromNode, toNode, alreadyVisited) {
|
|
if (toNode.id) {
|
|
// If an element with an ID is being morphed then it is will be in the final
|
|
// DOM so clear it out of the saved elements collection
|
|
delete savedEls[toNode.id];
|
|
}
|
|
|
|
if (onBeforeMorphEl(fromNode, toNode) === false) {
|
|
return;
|
|
}
|
|
|
|
morphAttrs(fromNode, toNode);
|
|
|
|
if (onBeforeMorphElChildren(fromNode, toNode) === false) {
|
|
return;
|
|
}
|
|
|
|
var curToNodeChild = toNode.firstChild;
|
|
var curFromNodeChild = fromNode.firstChild;
|
|
var curToNodeId;
|
|
|
|
var fromNextSibling;
|
|
var toNextSibling;
|
|
var savedEl;
|
|
var unmatchedEl;
|
|
|
|
outer: while(curToNodeChild) {
|
|
toNextSibling = curToNodeChild.nextSibling;
|
|
curToNodeId = curToNodeChild.id;
|
|
|
|
while(curFromNodeChild) {
|
|
var curFromNodeId = curFromNodeChild.id;
|
|
fromNextSibling = curFromNodeChild.nextSibling;
|
|
|
|
if (!alreadyVisited) {
|
|
if (curFromNodeId && (unmatchedEl = unmatchedEls[curFromNodeId])) {
|
|
unmatchedEl.parentNode.replaceChild(curFromNodeChild, unmatchedEl);
|
|
morphEl(curFromNodeChild, unmatchedEl, alreadyVisited);
|
|
curFromNodeChild = fromNextSibling;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
var curFromNodeType = curFromNodeChild.nodeType;
|
|
|
|
if (curFromNodeType === curToNodeChild.nodeType) {
|
|
var isCompatible = false;
|
|
|
|
if (curFromNodeType === 1) { // Both nodes being compared are Element nodes
|
|
if (curFromNodeChild.tagName === curToNodeChild.tagName) {
|
|
// We have compatible DOM elements
|
|
if (curFromNodeId || curToNodeId) {
|
|
// If either DOM element has an ID then we handle
|
|
// those differently since we want to match up
|
|
// by ID
|
|
if (curToNodeId === curFromNodeId) {
|
|
isCompatible = true;
|
|
}
|
|
} else {
|
|
isCompatible = true;
|
|
}
|
|
}
|
|
|
|
if (isCompatible) {
|
|
// We found compatible DOM elements so add a
|
|
// task to morph the compatible DOM elements
|
|
morphEl(curFromNodeChild, curToNodeChild, alreadyVisited);
|
|
}
|
|
} else if (curFromNodeType === 3) { // Both nodes being compared are Text nodes
|
|
isCompatible = true;
|
|
curFromNodeChild.nodeValue = curToNodeChild.nodeValue;
|
|
}
|
|
|
|
if (isCompatible) {
|
|
curToNodeChild = toNextSibling;
|
|
curFromNodeChild = fromNextSibling;
|
|
continue outer;
|
|
}
|
|
}
|
|
|
|
// No compatible match so remove the old node from the DOM
|
|
removeNode(curFromNodeChild, fromNode, alreadyVisited);
|
|
|
|
curFromNodeChild = fromNextSibling;
|
|
}
|
|
|
|
if (curToNodeId) {
|
|
if ((savedEl = savedEls[curToNodeId])) {
|
|
morphEl(savedEl, curToNodeChild, true);
|
|
curToNodeChild = savedEl; // We want to append the saved element instead
|
|
} else {
|
|
// The current DOM element in the target tree has an ID
|
|
// but we did not find a match in any of the corresponding
|
|
// siblings. We just put the target element in the old DOM tree
|
|
// but if we later find an element in the old DOM tree that has
|
|
// a matching ID then we will replace the target element
|
|
// with the corresponding old element and morph the old element
|
|
unmatchedEls[curToNodeId] = curToNodeChild;
|
|
}
|
|
}
|
|
|
|
// If we got this far then we did not find a candidate match for our "to node"
|
|
// and we exhausted all of the children "from" nodes. Therefore, we will just
|
|
// append the current "to node" to the end
|
|
fromNode.appendChild(curToNodeChild);
|
|
|
|
curToNodeChild = toNextSibling;
|
|
curFromNodeChild = fromNextSibling;
|
|
}
|
|
|
|
// We have processed all of the "to nodes". If curFromNodeChild is non-null then
|
|
// we still have some from nodes left over that need to be removed
|
|
while(curFromNodeChild) {
|
|
fromNextSibling = curFromNodeChild.nextSibling;
|
|
removeNode(curFromNodeChild, fromNode, alreadyVisited);
|
|
curFromNodeChild = fromNextSibling;
|
|
}
|
|
|
|
var specialElHandler = specialElHandlers[fromNode.tagName];
|
|
if (specialElHandler) {
|
|
specialElHandler(fromNode, toNode);
|
|
}
|
|
}
|
|
|
|
var morphedNode = fromNode;
|
|
var morphedNodeType = morphedNode.nodeType;
|
|
var toNodeType = toNode.nodeType;
|
|
|
|
// Handle the case where we are given two DOM nodes that are not
|
|
// compatible (e.g. <div> --> <span> or <div> --> TEXT)
|
|
if (morphedNodeType === 1) {
|
|
if (toNodeType === 1) {
|
|
if (morphedNode.tagName !== toNode.tagName) {
|
|
onNodeDiscarded(fromNode);
|
|
morphedNode = moveChildren(morphedNode, document.createElement(toNode.tagName));
|
|
}
|
|
} else {
|
|
// Going from an element node to a text node
|
|
return toNode;
|
|
}
|
|
} else if (morphedNodeType === 3) { // Text node
|
|
if (toNodeType === 3) {
|
|
morphedNode.nodeValue = toNode.nodeValue;
|
|
return morphedNode;
|
|
} else {
|
|
onNodeDiscarded(fromNode);
|
|
// Text node to something else
|
|
return toNode;
|
|
}
|
|
}
|
|
|
|
morphEl(morphedNode, toNode, false);
|
|
|
|
// Fire the "onNodeDiscarded" event for any saved elements
|
|
// that never found a new home in the morphed DOM
|
|
for (var savedElId in savedEls) {
|
|
if (savedEls.hasOwnProperty(savedElId)) {
|
|
var savedEl = savedEls[savedElId];
|
|
onNodeDiscarded(savedEl);
|
|
walkDiscardedChildNodes(savedEl);
|
|
}
|
|
}
|
|
|
|
if (morphedNode !== fromNode && fromNode.parentNode) {
|
|
fromNode.parentNode.replaceChild(morphedNode, fromNode);
|
|
}
|
|
|
|
return morphedNode;
|
|
}
|
|
|
|
module.exports = morphdom;
|
|
},{}]},{},[1])(1)
|
|
}); |