summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authormconway <michael.c.conway@gmail.com>2012-04-13 12:30:16 (GMT)
committer mconway <michael.c.conway@gmail.com>2012-04-13 12:30:16 (GMT)
commitae4ac981d4314359df5e60f6516d7f18e4d437bc (patch)
treee74667ec5905a3d4d1cf55cdb6bf47959c28eb9c
parentcfa179dab9db682963d850a9fce7a8d5c6f30ebc (diff)
downloadQCG-Data-ae4ac981d4314359df5e60f6516d7f18e4d437bc.zip
QCG-Data-ae4ac981d4314359df5e60f6516d7f18e4d437bc.tar.gz
QCG-Data-ae4ac981d4314359df5e60f6516d7f18e4d437bc.tar.bz2
[#725] version 1.0.0 release activities
-rw-r--r--idrop-lite/pom.xml2
-rw-r--r--idrop-lite/src/main/java/org/irods/jargon/idrop/lite/Version.java2
-rw-r--r--idrop-swing/src/main/java/org/irods/jargon/idrop/desktop/systraygui/LoginDialog.java48
-rw-r--r--idrop-swing/src/main/resources/idrop.properties4
-rw-r--r--idrop-web/application.properties2
-rw-r--r--idrop-web/grails-app/conf/BuildConfig.groovy10
-rw-r--r--idrop-web/grails-app/conf/Config.groovy24
-rw-r--r--idrop-web/grails-app/views/common/_panelmessages.gsp3
-rw-r--r--idrop-web/grails-app/views/home/index.gsp1
-rw-r--r--idrop-web/web-app/js/jstree.checkbox.js187
-rw-r--r--idrop-web/web-app/js/jstree.contextmenu.js143
-rw-r--r--idrop-web/web-app/js/jstree.core.js1965
-rw-r--r--idrop-web/web-app/js/jstree.dnd.js162
-rw-r--r--idrop-web/web-app/js/jstree.hotkeys.js138
-rw-r--r--idrop-web/web-app/js/jstree.html.js69
-rw-r--r--idrop-web/web-app/js/jstree.json.js99
-rw-r--r--idrop-web/web-app/js/jstree.sort.js38
-rw-r--r--idrop-web/web-app/js/jstree.state.js39
-rw-r--r--idrop-web/web-app/js/jstree.themes.js215
-rw-r--r--idrop-web/web-app/js/jstree.ui.js201
-rw-r--r--idrop-web/web-app/js/jstree.unique.js33
-rw-r--r--idrop-web/web-app/js/jstree.xml.js185
-rw-r--r--idrop-web/web-app/js/vakata.js1930
-rw-r--r--pom.xml2
-rw-r--r--release_notes.txt40
25 files changed, 5506 insertions, 36 deletions
diff --git a/idrop-lite/pom.xml b/idrop-lite/pom.xml
index 587d0bb..1535c07 100644
--- a/idrop-lite/pom.xml
+++ b/idrop-lite/pom.xml
@@ -12,7 +12,7 @@
<name>idrop-lite</name>
<version>1.0.0-SNAPSHOT</version>
<properties>
- <jargon.version>3.0.0.2-SNAPSHOT</jargon.version>
+ <jargon.version>3.1.1</jargon.version>
</properties>
<dependencies>
<dependency>
diff --git a/idrop-lite/src/main/java/org/irods/jargon/idrop/lite/Version.java b/idrop-lite/src/main/java/org/irods/jargon/idrop/lite/Version.java
index 04c559c..17d1668 100644
--- a/idrop-lite/src/main/java/org/irods/jargon/idrop/lite/Version.java
+++ b/idrop-lite/src/main/java/org/irods/jargon/idrop/lite/Version.java
@@ -1,4 +1,4 @@
package org.irods.jargon.idrop.lite;
public final class Version {
- public static String VERSION="20120328-1247";
+ public static String VERSION="20120413-0812";
}
diff --git a/idrop-swing/src/main/java/org/irods/jargon/idrop/desktop/systraygui/LoginDialog.java b/idrop-swing/src/main/java/org/irods/jargon/idrop/desktop/systraygui/LoginDialog.java
index 5cfbaf0..e32eb26 100644
--- a/idrop-swing/src/main/java/org/irods/jargon/idrop/desktop/systraygui/LoginDialog.java
+++ b/idrop-swing/src/main/java/org/irods/jargon/idrop/desktop/systraygui/LoginDialog.java
@@ -2,6 +2,7 @@ package org.irods.jargon.idrop.desktop.systraygui;
import java.awt.Color;
import java.awt.event.ActionEvent;
+import java.util.Properties;
import java.util.logging.Level;
import java.util.logging.Logger;
@@ -23,7 +24,7 @@ import org.openide.util.Exceptions;
import org.slf4j.LoggerFactory;
/**
- *
+ *
* @author mikeconway
*/
public class LoginDialog extends JDialog {
@@ -53,8 +54,7 @@ public class LoginDialog extends JDialog {
}
- private void loginNormally(
- ) {
+ private void loginNormally() {
// predispose based on preferences
String host = idropCore.getIdropConfig().getPropertyForKey(IdropConfigurationService.ACCOUNT_CACHE_HOST);
txtHost.setText(host);
@@ -83,7 +83,7 @@ public class LoginDialog extends JDialog {
/**
* Action to take when login is initiated
- *
+ *
* @return
* @throws NumberFormatException
*/
@@ -125,16 +125,35 @@ public class LoginDialog extends JDialog {
StringBuilder sb = new StringBuilder();
final IRODSAccount irodsAccount;
try {
+ IdropPropertiesHelper helper = new IdropPropertiesHelper();
+ Properties classpathConfig = null;
+ try {
+ classpathConfig = helper.loadIdropProperties();
+ if (classpathConfig == null || classpathConfig.isEmpty()) {
+ throw new IdropRuntimeException("cannot find classpath idrop.properties to check for login preset");
+ }
+ } catch (IdropException ex) {
+ log.error("exception getting idrop.properties from the classpath", ex);
+ throw new IdropRuntimeException("cannot find classpath idrop.properties to check for login preset", ex);
+ }
+
+ String loginPreset = classpathConfig.getProperty(IdropPropertiesHelper.LOGIN_PRESET);
+ boolean useLoginPreset = false;
+
+ if (loginPreset != null) {
+ useLoginPreset = Boolean.valueOf(loginPreset);
+ }
+
// validated, now try to log in
- if (idropCore.getIdropConfig().isLoginPreset()) {
+ if (useLoginPreset) {
log.debug("creating account with presets");
- String presetHost = idropCore.getIdropConfig().getIdropProperties().getProperty(IdropPropertiesHelper.LOGIN_PRESET_HOST);
+ String presetHost = classpathConfig.getProperty(IdropPropertiesHelper.LOGIN_PRESET_HOST);
log.info("presetHost:{}", presetHost);
- int presetPort = Integer.parseInt(idropCore.getIdropConfig().getIdropProperties().getProperty(IdropPropertiesHelper.LOGIN_PRESET_PORT));
+ int presetPort = Integer.parseInt(classpathConfig.getProperty(IdropPropertiesHelper.LOGIN_PRESET_PORT));
log.info("presetPort:{}", presetPort);
- String presetZone = idropCore.getIdropConfig().getIdropProperties().getProperty(IdropPropertiesHelper.LOGIN_PRESET_ZONE);
+ String presetZone =classpathConfig.getProperty(IdropPropertiesHelper.LOGIN_PRESET_ZONE);
log.info("presetZone:{}", presetZone);
- String presetResource = idropCore.getIdropConfig().getIdropProperties().getProperty(
+ String presetResource = classpathConfig.getProperty(
IdropPropertiesHelper.LOGIN_PRESET_RESOURCE);
log.info("presetResource:{}", presetResource);
sb.append('/');
@@ -165,8 +184,10 @@ public class LoginDialog extends JDialog {
IRODSFileSystem irodsFileSystem = null;
- /* getting userAO will attempt the login */
-
+ /*
+ * getting userAO will attempt the login
+ */
+
try {
irodsFileSystem = idropCore.getIrodsFileSystem();
final UserAO userAO = irodsFileSystem.getIRODSAccessObjectFactory().getUserAO(irodsAccount);
@@ -232,9 +253,8 @@ public class LoginDialog extends JDialog {
}
/**
- * This method is called from within the constructor to initialize the form.
- * WARNING: Do NOT modify this code. The content of this method is always
- * regenerated by the Form Editor.
+ * This method is called from within the constructor to initialize the form. WARNING: Do NOT
+ * modify this code. The content of this method is always regenerated by the Form Editor.
*/
@SuppressWarnings("unchecked")
// <editor-fold defaultstate="collapsed"
diff --git a/idrop-swing/src/main/resources/idrop.properties b/idrop-swing/src/main/resources/idrop.properties
index ce5d671..213d6a5 100644
--- a/idrop-swing/src/main/resources/idrop.properties
+++ b/idrop-swing/src/main/resources/idrop.properties
@@ -10,10 +10,6 @@ idrop.show.gui=true
# swing look and feel (Nimbus, GTK, System, Metal, Motif)
idrop.lookandfeel=System
transfer.database=transferDatabase
-#login.preset.host=diamond.ils.unc.edu
-#login.preset.zone=lifelibZone
-#login.preset.resource=lifelibResc1
-#login.preset.port=2247
device.name=
transferengine.record.successful.files=true
diff --git a/idrop-web/application.properties b/idrop-web/application.properties
index a28ed9b..5e50e98 100644
--- a/idrop-web/application.properties
+++ b/idrop-web/application.properties
@@ -3,7 +3,7 @@
app.grails.version=2.0.1
app.name=idrop-web
app.servlet.version=2.5
-app.version=1.0.0-SNAPSHOT
+app.version=1.0.0
plugins.hibernate=2.0.1
plugins.logging=0.1
plugins.tomcat=2.0.1
diff --git a/idrop-web/grails-app/conf/BuildConfig.groovy b/idrop-web/grails-app/conf/BuildConfig.groovy
index 7818e23..860a591 100644
--- a/idrop-web/grails-app/conf/BuildConfig.groovy
+++ b/idrop-web/grails-app/conf/BuildConfig.groovy
@@ -25,14 +25,14 @@ grails.project.dependency.resolution = {
}
dependencies {
// specify dependencies here under either 'build', 'compile', 'runtime', 'test' or 'provided' scopes eg.
- test 'org.irods.jargon:jargon-test:3.1.1-SNAPSHOT'
+ test 'org.irods.jargon:jargon-test:3.1.1'
test 'org.mockito:mockito-all:1.8.1'
compile 'commons-io:commons-io:2.1'
provided 'junit:junit:4.8.1'
- compile 'org.irods.jargon:jargon-core:3.1.1-SNAPSHOT'
- compile 'org.irods.jargon:jargon-security:3.1.1-SNAPSHOT'
- compile 'org.irods.jargon:jargon-data-utils:3.1.1-SNAPSHOT'
- compile ('org.irods.jargon:jargon-user-tagging:3.1.1-SNAPSHOT') { exclude 'junit' }
+ compile 'org.irods.jargon:jargon-core:3.1.1'
+ compile 'org.irods.jargon:jargon-security:3.1.1'
+ compile 'org.irods.jargon:jargon-data-utils:3.1.1'
+ compile ('org.irods.jargon:jargon-user-tagging:3.1.1') { exclude 'junit' }
compile 'org.springframework.security:spring-security-core:3.0.5.RELEASE'
compile 'org.springframework.security:spring-security-web:3.0.5.RELEASE'
compile 'org.springframework.security:spring-security-config:3.0.5.RELEASE'
diff --git a/idrop-web/grails-app/conf/Config.groovy b/idrop-web/grails-app/conf/Config.groovy
index 42b7f6a..cf519b3 100644
--- a/idrop-web/grails-app/conf/Config.groovy
+++ b/idrop-web/grails-app/conf/Config.groovy
@@ -12,8 +12,9 @@
environments {
/*production { grails.serverURL = "http://lifetime-library.ils.unc.edu/${appName}" }
production { grails.serverURL = "http://iren-web.renci.org:8080/${appName}" }
+ production { grails.serverURL = "http://srbbrick15.ucsd.edu:1525//${appName}" }
production { grails.serverURL = "http://www.irods.org" } */
- production { grails.serverURL = "http://www.irods.org" }
+ production { grails.serverURL = "http://iren-web.renci.org:8080/${appName}" }
development { grails.serverURL = "http://localhost:8080/${appName}" }
test { grails.serverURL = "http://localhost:8080/${appName}" }
}
@@ -32,19 +33,32 @@ environments {
/*
* 3) iDROP web includes the idrop-lite Java applet, which is launched from the iDROP web interface. The interface needs to know where to find this jar file.
* The Jar file should be placed on a web server in an accessible directory, and configured below
- */
+ *
+ *
+idrop.config.idrop.lite.applet.jar="idrop-lite-1.0.0-SNAPSHOT-jar-with-dependencies.jar"
+idrop.config.idrop.lite.codebase="http://iren-web.renci.org/idrop-web/applet"
+idrop.config.idrop.lite.use.applet.dir=false
idrop.config.idrop.lite.applet.jar="idrop-lite-1.0.0-SNAPSHOT-jar-with-dependencies.jar"
+idrop.config.idrop.lite.codebase="https://lifetime-library.ils.unc.edu/llclient"
+idrop.config.idrop.lite.use.applet.dir=false
+ */
+
+idrop.config.idrop.lite.applet.jar="idrop-lite-1.0.0-jar-with-dependencies.jar"
idrop.config.idrop.lite.codebase="http://iren-web.renci.org/idrop-web/applet"
-idrop.config.idrop.jnlp="http://iren-web.renci.org:8080/idrop/idrop.jnlp"
idrop.config.idrop.lite.use.applet.dir=false
+
/*
* 4) iDROP web includes a link to launch the iDROP desktop GUI, using Java WebStart. WebStart looks for a jnlp file and the accompanying
* lib directories. This should be deployed on a web server at some accessible location and referred to here
+ *
+ * idrop.config.idrop.jnlp="http://iren-web.renci.org:8080/idrop/idrop.jnlp"
+ *
+ * idrop.config.idrop.jnlp="https://lifetime-library.ils.unc.edu/llclient/idrop.jnlp"
*/
-idrop.config.idrop.jnlp="http://iren-web.renci.org:8080/idrop/idrop.jnlp"
+idrop.config.idrop.jnlp="http://iren-web.renci.org:8080/idrop/idrop.jnlp"
/*
* Some properties may be set in an external configuration file, as configured below
*/
@@ -151,7 +165,7 @@ log4j = {
info 'org.irods.jargon'
warn 'org.irods.jargon.spring.security'
warn 'org.springframework'
- info 'grails.app'
+ warn 'grails.app'
warn 'org.mortbay.log',
'grails.app.controller',
diff --git a/idrop-web/grails-app/views/common/_panelmessages.gsp b/idrop-web/grails-app/views/common/_panelmessages.gsp
deleted file mode 100644
index 8a53243..0000000
--- a/idrop-web/grails-app/views/common/_panelmessages.gsp
+++ /dev/null
@@ -1,3 +0,0 @@
-<g:if test="${flash.message}">
-<div class="message">${flash.message}<!-- panel messages --></div>
-</g:if>
diff --git a/idrop-web/grails-app/views/home/index.gsp b/idrop-web/grails-app/views/home/index.gsp
index 9397ec0..4960f9e 100644
--- a/idrop-web/grails-app/views/home/index.gsp
+++ b/idrop-web/grails-app/views/home/index.gsp
@@ -4,7 +4,6 @@
<g:javascript library="mydrop/search" />
<g:javascript library="mydrop/metadata" />
</head>
-<g:render template="/common/panelmessages" />
<div id="tabs" class="wrapper"
style="height: 820px; position: relative;">
diff --git a/idrop-web/web-app/js/jstree.checkbox.js b/idrop-web/web-app/js/jstree.checkbox.js
new file mode 100644
index 0000000..96ebfb1
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.checkbox.js
@@ -0,0 +1,187 @@
+/* File: jstree.checkbox.js
+Adds checkboxes to the tree.
+*/
+(function ($) {
+ $.jstree.plugin("checkbox", {
+ __construct : function () {
+ this.get_container()
+ .bind("__construct.jstree", $.proxy(function () {
+ // TODO: on move/copy - clean new location and parents
+ }, this))
+ .bind("move_node.jstree, copy_node.jstree", function (e, data) {
+ if(data.rslt.old_instance && data.rslt.old_parent && $.isFunction(data.rslt.old_instance.checkbox_repair)) {
+ data.rslt.old_instance.checkbox_repair(data.rslt.old_parent);
+ }
+ if(data.rslt.new_instance && $.isFunction(data.rslt.new_instance.checkbox_repair)) {
+ data.rslt.new_instance.checkbox_repair(data.rslt.parent);
+ }
+ })
+ .bind("delete_node.jstree", function (e, data) {
+ this.checkbox_repair(data.rslt.parent);
+ })
+ .delegate("a", "click.jstree", $.proxy(function (e) {
+ e.preventDefault();
+ e.currentTarget.blur();
+ var obj = this.get_node(e.currentTarget);
+ this.toggle_check(obj);
+ }, this));
+ },
+ defaults : {
+ three_state : true
+ },
+ _fn : {
+ /*
+ Group: CHECKBOX functions
+ */
+ check_node : function (obj) {
+ obj = this.get_node(obj);
+ obj.find(' > a > .jstree-checkbox').removeClass('jstree-unchecked jstree-undetermined').addClass('jstree-checked').children(':checkbox').prop('checked', true).prop('undermined', false);
+ this.checkbox_repair(obj);
+ },
+ uncheck_node : function (obj) {
+ obj = this.get_node(obj);
+ obj.find(' > a > .jstree-checkbox').removeClass('jstree-checked jstree-undetermined').addClass('jstree-unchecked').children(':checkbox').prop('checked', false).prop('undermined', false);
+ this.checkbox_repair(obj);
+ },
+ toggle_check : function (obj) {
+ obj = obj.find(' > a > .jstree-checkbox').removeClass('jstree-undetermined').toggleClass('jstree-checked');
+ if(!obj.hasClass('jstree-checked')) {
+ obj.addClass('jstree-unchecked').children(':checkbox').prop('checked', false).prop('undermined', false);
+ }
+ else {
+ obj.children(':checkbox').prop('checked', true).prop('undermined', false);
+ }
+ this.checkbox_repair(this.get_node(obj));
+ },
+ uncheck_all : function (context) {
+ var ret = context ? $(context).find(".jstree-checked").closest('li') : this.get_container().find(".jstree-checked").closest('li');
+ ret.children(".jstree-checkbox").removeClass("jstree-checked jstree-undetermined").addClass('jstree-unchecked').children(':checkbox').prop('checked', false).prop('undermined', false);
+ this.__callback({ "obj" : ret });
+ },
+
+ checkbox_repair : function (obj) {
+ if(!this.get_settings(true).checkbox.three_state) { return false; }
+
+ if(!obj || obj === -1) {
+ obj = this.get_container_ul().children('li');
+ }
+ if(obj.length > 1) {
+ obj.each($.proxy(function (i, d) {
+ this.checkbox_repair($(d));
+ }, this));
+ }
+
+ var c = obj.find(' > a > .jstree-checkbox'),
+ fix_up = true,
+ p, st, sc, su, si;
+
+ if(!c.hasClass('jstree-checked') && !c.hasClass('jstree-unchecked')) {
+ p = this.get_parent(obj);
+ if(p && p !== -1 && p.length && p.find('> a > .jstree-checked')) { c.addClass('jstree-checked'); }
+ else { c.addClass('jstree-unchecked'); }
+ fix_up = false;
+ }
+
+ if(c.hasClass('jstree-checked')) {
+ obj.find('.jstree-checkbox').removeClass('jstree-undetermined jstree-unchecked').addClass('jstree-checked').children(':checkbox').prop('checked', true).prop('undermined', false);
+ }
+ if(c.hasClass('jstree-unchecked')) {
+ obj.find('.jstree-checkbox').removeClass('jstree-undetermined jstree-checked').addClass('jstree-unchecked').children(':checkbox').prop('checked', false).prop('undermined', false);
+ }
+
+ while(fix_up) {
+ obj = this.get_parent(obj);
+ if(!obj || obj === -1 || !obj.length) { return; }
+
+ st = obj.find(' > ul > li');
+ sc = st.find(' > a > .jstree-checked').length;
+ su = st.find(' > a > .jstree-unchecked').length;
+ si = st.find(' > a > .jstree-undetermined').length;
+ st = st.length;
+
+ if(sc + su + si < st) { return; }
+
+ if(su === st) {
+ c = obj.find(' > a > .jstree-checkbox');
+ if(c.hasClass('jstree-unchecked')) { return; }
+ c.removeClass('jstree-undetermined jstree-checked').addClass('jstree-unchecked').children(':checkbox').prop('checked', false).prop('undermined', false);
+ continue;
+ }
+ if(sc === st) {
+ c = obj.find(' > a > .jstree-checkbox');
+ if(c.hasClass('jstree-checked')) { return; }
+ c.removeClass('jstree-undetermined jstree-unchecked').addClass('jstree-checked').children(':checkbox').prop('checked', true).prop('undermined', false);
+ continue;
+ }
+ obj.parentsUntil(".jstree", "li").andSelf().find(' > a > .jstree-checkbox').removeClass('jstree-checked jstree-unchecked').addClass('jstree-undetermined').children(':checkbox').prop('checked', false).prop('undetermined', true);
+ return;
+ }
+ },
+
+ clean_node : function(obj) {
+ obj = this.__call_old();
+ var t = this;
+ obj = obj.each(function () {
+ var o = $(this),
+ d = o.data("jstree");
+ o.find(" > a > .jstree-checkbox").remove();
+ o.children("a").prepend("<ins class='jstree-icon jstree-checkbox " + (d && d.checkbox && d.checkbox.checked === true ? 'jstree-checked' : '') + ( (d && d.checkbox && d.checkbox.checked === false) || !t.get_settings(true).checkbox.three_state ? 'jstree-unchecked' : '') + " '><input class='jstree-check' type='checkbox' " + (d && d.checkbox && d.checkbox.checked ? ' checked="checked" ' : '') + " name='" + (d && d.checkbox && typeof d.checkbox.name !== 'undefined' ? d.checkbox.name : 'jstree[]') + "' value='" + (d && d.checkbox && typeof d.checkbox.value !== 'undefined' ? d.checkbox.value : o.attr('id')) + "' /></ins>");
+ });
+ t.checkbox_repair(obj);
+ return obj;
+ },
+ get_state : function () {
+ var state = this.__call_old();
+ state.checked = [];
+ this.get_container().find('.jstree-checked').closest('li').each(function () { if(this.id) { state.checked.push(this.id); } });
+ return state;
+ },
+ set_state : function (state, callback) {
+ if(this.__call_old()) {
+ if(state.checkbox) {
+ var _this = this;
+ this.uncheck_all();
+ $.each(state.checkbox, function (i, v) {
+ _this.check_node(document.getElementById(v));
+ });
+ this.checkbox_repair();
+ delete state.checkbox;
+ this.set_state(state, callback);
+ return false;
+ }
+ return true;
+ }
+ return false;
+ },
+ get_json : function (obj, is_callback) {
+ var r = this.__call_old(), i;
+ if(is_callback) {
+ i = obj.find('> a > ins > :checkbox');
+ r.data.jstree.checkbox = {};
+ r.data.jstree.checkbox.checked = i.parent().hasClass('jstree-checked');
+ if(i.attr('name') != 'jstree[]') { r.data.checkbox.name = i.attr('name'); }
+ if(i.val() != obj.attr('id')) { r.data.checkbox.value = i.val(); }
+ }
+ return r;
+ }
+ }
+ });
+ $(function () {
+ // add checkbox specific CSS
+ var css_string = '' +
+ '.jstree a > .jstree-checkbox { height:16px; width:16px; margin-right:1px; } ' +
+ '.jstree-rtl a > .jstree-checkbox { margin-right:0; margin-left:1px; } ' +
+ '.jstree .jstree-check { margin:0; padding:0; border:0; display:inline; vertical-align:text-bottom; } ';
+ // Correct IE 6 (does not support the > CSS selector)
+ if($.jstree.IS_IE6) {
+ css_string += '' +
+ '.jstree li a .jstree-checkbox { height:16px; width:16px; background:transparent; margin-right:1px; } ' +
+ '.jstree-rtl li a .jstree-checkbox { margin-right:0; margin-left:1px; } ';
+ }
+ // the default stylesheet
+ $.vakata.css.add_sheet({ str : css_string, title : "jstree" });
+ });
+ // include the checkbox plugin by default
+ $.jstree.defaults.plugins.push("checkbox");
+})(jQuery);
+//*/ \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.contextmenu.js b/idrop-web/web-app/js/jstree.contextmenu.js
new file mode 100644
index 0000000..f7cd28b
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.contextmenu.js
@@ -0,0 +1,143 @@
+/* File: jstree.contextmenu.js
+Enables a rightclick contextmenu.
+*/
+/* Group: jstree sort plugin */
+(function ($) {
+ $.jstree.plugin("contextmenu", {
+ __construct : function () {
+ this.get_container()
+ .delegate("a", "contextmenu.jstree", $.proxy(function (e) {
+ e.preventDefault();
+ if(!this.is_loading(e.currentTarget)) {
+ this.show_contextmenu(e.currentTarget, e.pageX, e.pageY);
+ }
+ }, this))
+ .delegate("a", "click.jstree", $.proxy(function (e) {
+ if(this.data.contextmenu.visible) {
+ this._hide_contextmenu();
+ }
+ }, this));
+ $(document).bind("context_hide.vakata", $.proxy(function () { this.data.contextmenu = false; }, this));
+ },
+ __destruct : function () {
+ if(this.data.contextmenu) {
+ this._hide_contextmenu();
+ }
+ },
+ defaults : {
+ select_node : true,
+ show_at_node : true,
+ items : function (o) { // Could be an object directly
+ return {
+ "create" : {
+ "separator_before" : false,
+ "separator_after" : true,
+ "label" : "Create",
+ "action" : function (data) {
+ var inst = $.jstree._reference(data.reference),
+ obj = inst.get_node(data.reference);
+ inst.create_node(obj);
+ inst.edit(obj);
+ }
+ },
+ "rename" : {
+ "separator_before" : false,
+ "separator_after" : false,
+ "label" : "Rename",
+ "action" : function (data) {
+ var inst = $.jstree._reference(data.reference),
+ obj = inst.get_node(data.reference);
+ inst.edit(obj);
+ }
+ },
+ "remove" : {
+ "separator_before" : false,
+ "icon" : false,
+ "separator_after" : false,
+ "label" : "Delete",
+ "action" : function (data) {
+ var inst = $.jstree._reference(data.reference),
+ obj = inst.get_node(data.reference);
+ inst.delete_node(obj);
+ }
+ },
+ "ccp" : {
+ "separator_before" : true,
+ "icon" : false,
+ "separator_after" : false,
+ "label" : "Edit",
+ "action" : false,
+ "submenu" : {
+ "cut" : {
+ "separator_before" : false,
+ "separator_after" : false,
+ "label" : "Cut",
+ "action" : function (data) {
+ var inst = $.jstree._reference(data.reference),
+ obj = inst.get_node(data.reference);
+ inst.cut(obj);
+ }
+ },
+ "copy" : {
+ "separator_before" : false,
+ "icon" : false,
+ "separator_after" : false,
+ "label" : "Copy",
+ "action" : function (data) {
+ var inst = $.jstree._reference(data.reference),
+ obj = inst.get_node(data.reference);
+ inst.copy(obj);
+ }
+ },
+ "paste" : {
+ "separator_before" : false,
+ "icon" : false,
+ "_disabled" : !(this.can_paste()),
+ "separator_after" : false,
+ "label" : "Paste",
+ "action" : function (data) {
+ var inst = $.jstree._reference(data.reference),
+ obj = inst.get_node(data.reference);
+ inst.paste(obj);
+ }
+ }
+ }
+ }
+ };
+ }
+ },
+ _fn : {
+ show_contextmenu : function (obj, x, y) {
+ obj = this.get_node(obj);
+ var s = this.get_settings().contextmenu,
+ a = obj.children("a:visible:eq(0)"),
+ o = false,
+ i = false;
+ if(s.show_at_node || typeof x === "undefined" || typeof y === "undefined") {
+ o = a.offset();
+ x = o.left;
+ y = o.top + this.data.core.li_height;
+ }
+ if(s.select_node && this.data.ui && !this.is_selected(obj)) {
+ this.deselect_all();
+ this.select_node(obj);
+ }
+
+ i = obj.data("jstree") && obj.data("jstree").contextmenu ? obj.data("jstree").contextmenu : s.items;
+ if($.isFunction(i)) { i = i.call(this, obj); }
+
+ $(document).one("context_show.vakata", $.proxy(function (e, data) {
+ var cls = 'jstree-contextmenu';
+ if(this.data.themes.theme) {
+ cls += ' jstree-' + this.data.themes.theme + '-contextmenu';
+ }
+ $(data.element).addClass(cls);
+ }, this));
+ this.data.contextmenu.visible = true;
+ $.vakata.context.show(a, { 'x' : x, 'y' : y }, i);
+ this.__callback({ "obj" : obj, "x" : x, "y" : y });
+ }
+ }
+ });
+ $.jstree.defaults.plugins.push("contextmenu");
+})(jQuery); \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.core.js b/idrop-web/web-app/js/jstree.core.js
new file mode 100644
index 0000000..ca89faf
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.core.js
@@ -0,0 +1,1965 @@
+/*
+ * jsTree 1.0
+ * http://jstree.com/
+ *
+ * Copyright (c) 2011 Ivan Bozhanov (vakata.com)
+ *
+ * Licensed same as jquery - under the terms of either the MIT License or the GPL Version 2 License
+ * http://www.opensource.org/licenses/mit-license.php
+ * http://www.gnu.org/licenses/gpl.html
+ *
+ */
+
+/*jslint browser: true, onevar: true, undef: true, bitwise: true, strict: true */
+/*global window : false, clearInterval: false, clearTimeout: false, document: false, setInterval: false, setTimeout: false, jQuery: false, navigator: false, XSLTProcessor: false, DOMParser: false, XMLSerializer: false*/
+
+/* File: jstree.core.js
+The only required part of jstree it consists of a few functions bound to the $.jstree object, the actual plugin function and a few core functions for manipulating a tree.
+*/
+"use strict";
+(function () {
+ if(!jQuery) { throw "jsTree: jQuery not included."; }
+ if(jQuery.jstree) { return; } // prevent another load? maybe there is a better way?
+
+/* Group: $.jstree.
+Some static functions and variables, unless you know exactly what you are doing do not use these, but <$().jstree> instead.
+*/
+(function ($) {
+ var instances = [],
+ focused_instance = -1,
+ plugins = {},
+ functions = {};
+ /*
+ Variable: $.jstree
+ *object* Contains all static functions and variables used by jstree, some plugins also append variables.
+ */
+ $.jstree = {
+ /*
+ Variable: $.jstree.VERSION
+ *string* the version of jstree
+ */
+ VERSION : '1.0',
+
+ /*
+ Variable: $.jstree.IS_IE6
+ *boolean* indicating if the client is running Internet Explorer 6
+ */
+ IS_IE6 : (jQuery.browser.msie && parseInt(jQuery.browser.version,10) === 6),
+
+ /*
+ Variable: $.jstree.IS_IE7
+ *boolean* indicating if the client is running Internet Explorer 7
+ */
+ IS_IE7 : (jQuery.browser.msie && parseInt(jQuery.browser.version,10) === 6),
+
+ /*
+ Variable: $.jstree.IS_FF2
+ *boolean* indicating if the client is running Firefox 2
+ */
+ IS_FF2 : (jQuery.browser.mozilla && parseFloat(jQuery.browser.version,10) < 1.9),
+
+ /*
+ Function: $.jstree.__construct
+ Creates a new jstree instance, any arguments after the first one are merged and used to configure the tree.
+
+ `.data("jstree")` is also called on the container and is used for configuration (keep in mind you can specify this data using a "data-jstree" attribute)
+
+ Parameters:
+ container - *mixed* the container of the tree (this should not be the UL node, but a wrapper) - DOM node, jQuery object or selector
+ */
+ __construct : function (container) {
+ var s = {}, // settings
+ d = {}, // data
+ p = [], // plugins
+ t = [], // plugins temp
+ i = 0; // index
+ container = $(container);
+ if($.jstree._reference(container)) { $.jstree.__destruct(container); }
+ $.extend.apply(null, [true, s].concat(Array.prototype.slice.call(arguments, 1), (container.data("jstree") || {}) ));
+ p = $.isArray(s.plugins) ? s.plugins : $.jstree.defaults.plugins.slice();
+ p = $.vakata.array_unique(p);
+ s = $.extend(true, {}, $.jstree.defaults, s);
+ $.each(plugins, function (i, val) {
+ if(i !== "core" && $.inArray(i, p) === -1) { s[i] = null; delete s[i]; }
+ else { t.push(i); d[i] = {}; }
+ });
+ s.plugins = t;
+ i = parseInt(instances.push({}),10) - 1;
+ container
+ .data("jstree_instance_id", i)
+ .addClass("jstree jstree-" + i);
+
+ this.data = d;
+ this.get_index = function () { return i; };
+ this.get_container = function () { return container; };
+ this.get_container_ul = function () { return container.children("ul:eq(0)"); };
+ this.get_settings = function (writable) { return writable ? s : $.extend(true, {}, s); };
+ this.__trigger = function (ev, data) {
+ if(!ev) { return; }
+ if(!data) { data = {}; }
+ if(typeof ev === "string") { ev = ev.replace(".jstree","") + ".jstree"; }
+ data.inst = this;
+ this.get_container().triggerHandler(ev, data);
+ };
+ instances[i] = this;
+ $.each(t, function (j, val) { if(plugins[val]) { plugins[val].__construct.apply(instances[i]); } });
+ this.__trigger("__construct");
+ $.jstree._focus(i);
+ return this;
+ },
+ /*
+ Group: $.jstree.
+
+ Function: $.jstree.__destruct
+ Destroys an instance, and also clears `jstree-` prefixed classes and all events in the `jstree` namespace
+
+ Parameters:
+ instance - *mixed* the instance to destroy (this argument is passed to <$.jstree._reference> to get the instance)
+
+ See also:
+ <$.jstree._reference>
+ */
+ __destruct : function (instance) {
+ instance = $.jstree._reference(instance);
+ if(!instance) { return false; }
+ var s = instance.get_settings(),
+ n = instance.get_index(),
+ i = 0;
+ if(focused_instance === n) {
+ for(i in instances) {
+ if(instances.hasOwnProperty(i) && i != n) {
+ $.jstree._focus(i);
+ break;
+ }
+ }
+ if(focused_instance === n) { $.jstree._focus(false); }
+ }
+ $.each(s.plugins, function (i, val) {
+ try { plugins[val].__destruct.apply(instance); } catch(err) { }
+ });
+ instance.__trigger("__destruct");
+ instance.get_container()
+ .unbind(".jstree")
+ .undelegate(".jstree")
+ .removeData("jstree_instance_id")
+ .find("[class^='jstree']")
+ .andSelf()
+ .attr("class", function () { return this.className.replace(/jstree[^ ]*|$/ig,''); });
+ $(document)
+ .unbind(".jstree-" + n)
+ .undelegate(".jstree-" + n);
+ delete instances[n];
+ return true;
+ },
+ /*
+ Function: $.jstree.__call
+ Call a function on the instance and return the result
+
+ Parameters:
+ instance - *mixed* the instance to destroy (this argument is passed to <$.jstree._reference> to get the instance)
+ operation - *string* the operation to execute
+ args - *array* the arguments to pass to the function
+
+ See also:
+ <$.jstree._reference>
+ */
+ __call : function (instance, operation, args) {
+ instance = $.jstree._reference(instance);
+ if(!instance || !$.isFunction(instance[operation])) { return; }
+ return instance[operation].apply(instance, args);
+ },
+ /*
+ Function: $.jstree._reference
+ Returns an instance
+
+ Parameters:
+ needle - *mixed* - integer, DOM node contained inside a jstree container, ID string, jQuery object, selector
+ */
+ _reference : function (needle) {
+ if(instances[needle]) { return instances[needle]; }
+ var o = $(needle);
+ if(!o.length && typeof needle === "string") { o = $("#" + needle); }
+ if(!o.length) { return null; }
+ return instances[o.closest(".jstree").data("jstree_instance_id")] || null;
+ },
+ /*
+ Function: $.jstree._focused
+ Returns the currently focused instance (by default once an instance is created it is focused)
+ */
+ _focused : function () {
+ return instances[focused_instance] || null;
+ },
+ /*
+ Function: $.jstree._focus
+ Make an instance focused (which defocuses the previously focused instance)
+
+ Parameters:
+ instance - *mixed* the instance to focus (this argument is passed to <$.jstree._reference> to get the instance)
+
+ See also:
+ <$.jstree._reference>
+ */
+ _focus : function (instance) {
+ if(instance === false) {
+ instances[focused_instance].get_container().removeClass("jstree-focused");
+ instances[focused_instance].__trigger("_defocus");
+ focused_instance = -1;
+ return false;
+ }
+ instance = $.jstree._reference(instance);
+ if(!instance || instance.get_index() === focused_instance) { return false; }
+ if(focused_instance !== -1) {
+ instances[focused_instance].get_container().removeClass("jstree-focused");
+ instances[focused_instance].__trigger("_defocus");
+ }
+ focused_instance = instance.get_index();
+ instance.get_container().addClass("jstree-focused");
+ instance.__trigger("_focus");
+ return true;
+ },
+ /*
+ Function: $.jstree.plugin
+ Register a plugin
+
+ Parameters:
+ plugin_name - *string* the name of the new plugin (it will be used as a key in an object - make sure it is valid)
+ plugin_data - *object* consists of 4 keys. Default is:
+ >{
+ > __construct : $.noop, // this function will be executed when a new instance is created
+ > __destuct : $.noop, // this function will be executed when an instance is destroyed
+ > _fn : { }, // each key of this object should be a function that will extend the jstree prototype
+ > defaults : false // the default configuration for the plugin (if any)
+ >}
+ */
+ plugin : function (plugin_name, plugin_data) {
+ plugin_data = $.extend({}, {
+ __construct : $.noop,
+ __destuct : $.noop,
+ _fn : { },
+ defaults : false
+ }, plugin_data);
+ plugins[plugin_name] = plugin_data;
+ $.jstree.defaults[plugin_name] = plugin_data.defaults;
+ $.each(plugin_data._fn, function (i, val) {
+ val.plugin = plugin_name;
+ val.old = functions[i];
+ functions[i] = function () {
+ var rslt,
+ func = val,
+ args = Array.prototype.slice.call(arguments),
+ evnt = new $.Event("before.jstree"),
+ plgn = this.get_settings(true).plugins;
+
+ do {
+ if(func && func.plugin && $.inArray(func.plugin, plgn) !== -1) { break; }
+ func = func.old;
+ } while(func);
+ if(!func) { return; }
+
+ if(i.indexOf("_") === 0) {
+ rslt = func.apply(this, args);
+ }
+ else {
+ rslt = this.__trigger(evnt, { "func" : i, "args" : args, "plugin" : func.plugin });
+ if(rslt === false) { return; }
+ rslt = func.apply(
+ $.extend({}, this, {
+ __callback : function (data) {
+ this.__trigger( i, { "args" : args, "rslt" : data, "plugin" : func.plugin });
+ return data;
+ },
+ __call_old : function (replace_arguments) {
+ return func.old.apply(this, (replace_arguments ? Array.prototype.slice.call(arguments, 1) : args ) );
+ }
+ }), args);
+ }
+ return rslt;
+ };
+ functions[i].old = val.old;
+ functions[i].plugin = plugin_name;
+ });
+ },
+ /*
+ Variable: $.jstree.defaults
+ *object* storing all the default configuration options for every plugin and the core.
+ If this is modified all instances created after the modification, which do not explicitly specify some other value will use the new default.
+
+ Example:
+ >// this instance will use the _default_ theme
+ >$("#div0").jstree({ plugins : ["themes"] });
+ >$.jstree.defaults.themes.theme = "classic";
+ >// this instance will use the _classic_ theme
+ >$("#div1").jstree({ plugins : ["themes"] });
+ >// this instance will use the _apple_ theme
+ >$("#div2").jstree({ themes : { "theme" : "apple" }, plugins : ["themes"] });
+ */
+ defaults : {
+ plugins : []
+ }
+ };
+ /* Group: $().jstree()
+ The actual plugin wrapper, use this to create instances or execute functions on created instances.
+
+ Function: $().jstree
+
+ Creates an instance using the specified objects for containers, or executes a command on an instance, specified by a container.
+
+ Parameters:
+ settings - *mixed*
+
+ - if you pass an *object* a new instance will be created (using <$.jstree.__construct>)
+ for each of the objects in the jquery collection,
+ if an instance already exists on the container it will be destroyed first
+
+ - if you pass a *string* it will be executed using <$.jstree.__call> on each instance
+
+ Examples:
+ > // this creates an instance
+ > $("#some-id").jstree({
+ > plugins : [ "html_data", "themes", "ui" ]
+ > });
+ >
+ > // this executes a function on the instance
+ > $("#some-id").jstree("select_node", "#the-id-to-select");
+
+ See also:
+ <$.jstree.__construct>,
+ <$.jstree.__destruct>,
+ <$.jstree.__call>
+ */
+ $.fn.jstree = function (settings) {
+ var _is_method = (typeof settings == 'string'),
+ _arguments = Array.prototype.slice.call(arguments, 1),
+ _return = this;
+ this.each(function () {
+ if(_is_method) {
+ var val = $.jstree.__call(this, settings, _arguments);
+ if(typeof val !== "undefined" && (settings.indexOf("is_" === 0) || (val !== true && val !== false))) {
+ _return = val;
+ return false;
+ }
+ }
+ else {
+ _is_method = new $.jstree.__construct(this, settings);
+ }
+ });
+ return _return;
+ };
+ functions = $.jstree.__construct.prototype;
+})(jQuery);
+//*/
+
+(function ($) {
+ var ccp_node = false,
+ ccp_mode = false;
+
+ $(function() { $.jstree.SCROLLBAR_WIDTH = $.vakata.get_scrollbar_width(); });
+
+ $.jstree.plugin("core", {
+ __construct : function () {
+ this.data.core.rtl = (this.get_container().css("direction") === "rtl");
+ if(this.data.core.rtl) { this.get_container().addClass("jstree-rtl"); }
+ this.data.core.ready = false;
+
+ this.get_container()
+ .bind("__construct.jstree", $.proxy(function () {
+ // defer, so that events bound AFTER creating the instance (like __ready) are still handled
+ setTimeout($.proxy(function () { if(this) { this.init(); } }, this), 0);
+ }, this))
+ .bind("before.jstree", $.proxy(function (e, data) {
+ if(!/^is_locked|unlock$/.test(data.func) && this.data.core.locked) {
+ e.stopImmediatePropagation();
+ return false;
+ }
+ }, this))
+ .bind("create_node.jstree", $.proxy(function (e, data) {
+ this.clean_node(data.rslt.obj);
+ }, this))
+ .bind("load_node.jstree", $.proxy(function (e, data) {
+ // data.rslt.status
+ this.clean_node(data.rslt.obj === -1 ? this.get_container_ul().children('li') : data.rslt.obj.find('> ul > li'));
+ if(!this.data.core.ready && !this.get_container_ul().find('.jstree-loading:eq(0)').length) {
+ this.data.core.ready = true;
+ this.__trigger("__ready");
+ }
+ }, this))
+ .bind("__loaded.jstree", $.proxy(function (e, data) {
+ data.inst.get_container_ul().children('li').each(function () {
+ data.inst.correct_node(this);
+ });
+ }, this))
+ .bind("open_node.jstree", $.proxy(function (e, data) {
+ data.rslt.obj.find('> ul > li').each(function () {
+ data.inst.correct_node(this);
+ });
+ }, this))
+ .bind("mousedown.jstree", $.proxy(function () {
+ $.jstree._focus(this.get_index());
+ }, this))
+ .bind("dblclick.jstree", function () {
+ if(document.selection && document.selection.empty) { document.selection.empty(); }
+ else { if(window.getSelection) { var sel = window.getSelection(); try { sel.removeAllRanges(); sel.collapse(); } catch (er) { } } }
+ })
+ .delegate("li > ins", "click.jstree", $.proxy(function (e) {
+ // var trgt = $(e.target);
+ // if(trgt.is("ins") && e.pageY - trgt.offset().top < this.data.core.li_height) { this.toggle_node(trgt); }
+ this.toggle_node(e.target);
+ }, this));
+ },
+ __destruct : function () {
+
+ },
+ /* Class: jstree */
+ /*
+ Variable: data
+ *object* Provides storage for plugins (aside from private variables). Every plugin has an key in this object.
+ > this.data.<plugin_name>;
+ This is useful for detecting if some plugin is included in the instance (plugins also use this for dependencies and enhancements).
+
+ Function: get_index
+ Returns an *integer*, which is the instance's index. Every instance on the page has an unique index, when destroying an intance the index will not be reused.
+
+ Function: get_container
+ Returns the jQuery extended container of the tree (the element you used when constructing the tree).
+
+ Function: get_container_ul
+ Returns the jQuery extended first UL node inside the container of the tree.
+
+ Function: get_settings
+ Returns the settings for the tree.
+
+ Parameters:
+ writable - *boolean* whether to return a copy of the settings object or a reference to it.
+
+ Example:
+ > $("#div1").jstree("get_settings"); // will return a copy
+ > $.jstree._reference("#div1").get_settings(); // same as above
+ > $.jstree._focused().get_settings(true); // a reference. BE CAREFUL!
+
+ Function: __trigger
+ Used internally to trigger events on the container node.
+
+ Parameters:
+ event_name - the name of the event to trigger (the *jstree* namespace will be appended to it)
+ data - the additional object to pass along with the event. By default _data.inst_ will be the current instance, so when you bind to the event, you can access the instance easily.
+ > $("div").bind("some-event.jstree", function (e, data) { data.inst.some_function(); });
+ */
+ /*
+ Group: CORE options
+
+ Variable: config.core.strings
+ *mixed* used to store all localization strings. Default is _false_.
+
+ Example 1:
+ >$("div").jstree({
+ > core : {
+ > strings : function (s) {
+ > if(s === "Loading ...") { s = "Please wait ..."; }
+ > return s;
+ > }
+ > }
+ >});
+
+ Example 2:
+ >$("div").jstree({
+ > core : {
+ > strings : {
+ > "Loading ..." : "Please wait ..."
+ > }
+ > }
+ >});
+
+ See also:
+ <_get_string>
+ */
+ defaults : {
+ strings : false
+ },
+ _fn : {
+ /*
+ Group: CORE functions
+
+ Function: _get_string
+ Used to get the common string in the tree.
+
+ If <config.core.strings> is set to a function, that function is called with a single parameter (the needed string), the response is returned.
+
+ If <config.core.strings> is set to an object, the key named as the needed string is returned.
+
+ If <config.core.strings> is not set, the the needed string is returned.
+
+ Parameters:
+ needed_string - *string* the needed string
+ */
+ _get_string : function (s) {
+ var a = this.get_settings(true).core.strings;
+ if($.isFunction(a)) { return a.call(this, s); }
+ if(a && a[s]) { return a[s]; }
+ return s;
+ },
+ /*
+ Function: init
+ Used internally. This function is called once the core plugin is constructed.
+
+ Triggers:
+ <__loaded>
+
+ Event: __loaded
+ This event is triggered in the *jstree* namespace when data is first rendered in the tree. It won't be triggered after a refresh. Fires only once.
+
+ Parameters:
+ data.inst - the instance
+
+ Example:
+ > $("div").bind("__loaded.jstree", function (e, data) { data.inst.do_something(); });
+
+ Event: __ready
+ This event is triggered in the *jstree* namespace when all initial loading is done. It won't be triggered after a refresh. Fires only once.
+
+ Parameters:
+ data.inst - the instance
+ */
+ init : function () {
+ this.data.core.original_container_html = this.get_container().find(" > ul > li").clone(true);
+ this.data.core.original_container_html.find("li").andSelf().contents().filter(function() { return this.nodeType == 3 && (!this.nodeValue || /^\s+$/.test(this.nodeValue)); }).remove();
+ this.get_container().html("<ul><li class='jstree-loading'><a href='#'>" + this._get_string("Loading ...") + "</a></li></ul>");
+ this.clean_node(-1);
+ this.data.core.li_height = this.get_container_ul().children("li:eq(0)").height() || 18;
+ this.load_node(-1, function () {
+ this.__trigger("__loaded");
+ });
+ },
+ /*
+ Function: lock
+ Used to lock the tree. When the tree is in a locked state, no functions can be called on the instance (except <is_locked> and <unlock>).
+ Additionally a _jstree-locked_ class is applied on the container.
+
+ Triggers:
+ <lock>
+
+ Event: lock
+ This event is triggered in the *jstree* namespace when the tree is locked.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else).
+ data.rslt - _null_
+
+ Example:
+ > $("div").bind("lock.jstree", function (e, data) { data.inst.do_something(); });
+ */
+ lock : function () {
+ this.data.core.locked = true;
+ this.get_container().addClass("jstree-locked");
+ this.__callback();
+ },
+ /*
+ Function: unlock
+ Used to unlock the tree. Instance can be used normally again. The _jstree-locked_ class is removed from the container.
+
+ Triggers:
+ <unlock>
+
+ Event: unlock
+ This event is triggered in the *jstree* namespace when the tree is unlocked.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else).
+ data.rslt - _null_
+
+ Example:
+ > $("div").bind("unlock.jstree", function (e, data) { data.inst.do_something(); });
+ */
+ unlock : function () {
+ this.data.core.locked = false;
+ this.get_container().removeClass("jstree-locked");
+ this.__callback();
+ },
+ /*
+ Function: is_locked
+ Used to get the locked status of the tree.
+
+ Returns:
+ locked - *boolean* _true_ if tree is locked, _false_ otherwise
+ */
+ is_locked : function () {
+ return this.data.core.locked;
+ },
+ /*
+ Function: get_node
+ Get a hold of the LI node (which represents the jstree node).
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+
+ Returns:
+ jquery collection - node was found, the collection contains the LI node
+ -1 - the tree container was referenced
+ false - on failure (obj is not part of a tree, or does not exists in the DOM)
+ */
+ get_node : function (obj) {
+ var $obj = $(obj, this.get_container());
+ if($obj.is(".jstree") || obj == -1) { return -1; }
+ $obj = $obj.closest("li", this.get_container());
+ return $obj.length ? $obj : false;
+ },
+ /*
+ Function: get_next
+ Get the next sibling of a node
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+ strict - *boolean* if set to _true_ jstree will only return immediate siblings, otherwise, if _obj_ is the last child of its parent, the parent's next sibling is returned.
+
+ Returns:
+ jquery collection - node was found, the collection contains the LI node
+ -1 - the tree container was referenced
+ false - node was not found, or failure (obj is not part of a tree, or does not exists in the DOM)
+ */
+ get_next : function (obj, strict) {
+ obj = this.get_node(obj);
+ if(obj === -1) { return this.get_container_ul().children("li:eq(0)"); }
+ if(!obj || !obj.length) { return false; }
+ if(strict) { return (obj.nextAll("li").size() > 0) ? obj.nextAll("li:eq(0)") : false; }
+ if(obj.hasClass("jstree-open")) { return obj.find("li:eq(0)"); }
+ else if(obj.nextAll("li").size() > 0) { return obj.nextAll("li:eq(0)"); }
+ else { return obj.parentsUntil(".jstree","li").next("li").eq(0); }
+ },
+ /*
+ Function: get_prev
+ Get the previous sibling of a node
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+ strict - *boolean* if set to _true_ jstree will only return immediate siblings, otherwise, if _obj_ is the first child of its parent, the parent's previous sibling is returned.
+
+ Returns:
+ jquery collection - node was found, the collection contains the LI node
+ -1 - the tree container was referenced
+ false - node was not found, or failure (obj is not part of a tree, or does not exists in the DOM)
+ */
+ get_prev : function (obj, strict) {
+ obj = this.get_node(obj);
+ if(obj === -1) { return this.get_container().find("> ul > li:last-child"); }
+ if(!obj || !obj.length) { return false; }
+ if(strict) { return (obj.prevAll("li").length > 0) ? obj.prevAll("li:eq(0)") : false; }
+ if(obj.prev("li").length) {
+ obj = obj.prev("li").eq(0);
+ while(obj.hasClass("jstree-open")) { obj = obj.children("ul:eq(0)").children("li:last"); }
+ return obj;
+ }
+ else { var o = obj.parentsUntil(".jstree","li:eq(0)"); return o.length ? o : false; }
+ },
+ /*
+ Function: get_parent
+ Get the parent of a node
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+
+ Returns:
+ jquery collection - node was found, the collection contains the LI node
+ -1 - when _obj_ was a root node
+ false - on failure (obj is not part of a tree, or does not exists in the DOM)
+ */
+ get_parent : function (obj) {
+ obj = this.get_node(obj);
+ if(obj === -1 || !obj || !obj.length) { return false; }
+ var o = obj.parentsUntil(".jstree", "li:eq(0)");
+ return o.length ? o : -1;
+ },
+ /*
+ Function: get_children
+ Get all the children of a node
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc. If _-1_ is used all root nodes are returned.
+
+ Returns:
+ jquery collection - node was found, the collection contains the LI nodes of all immediate children
+ false - on failure (obj is not part of a tree, or does not exists in the DOM)
+ */
+ get_children : function (obj) {
+ obj = this.get_node(obj);
+ if(obj === -1) { return this.get_container_ul().children("li"); }
+ if(!obj || !obj.length) { return false; }
+ return obj.find("> ul > li");
+ },
+ /*
+ Function: is_parent
+ Check if a node is a parent.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+
+ Returns:
+ true - _obj_ has children or is closed (will be loaded)
+ false - _obj_ is not a valid node or has no children (leaf node)
+ */
+ is_parent : function (obj) { obj = this.get_node(obj); return obj && obj !== -1 && (obj.find("> ul > li:eq(0)").length || obj.hasClass("jstree-closed")); },
+ /*
+ Function: is_loaded
+ Check if a node is loaded.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+
+ Returns:
+ true - _obj_ has children or is leaf
+ false - _obj_ is currently loading or is not a leaf, but has no children
+ */
+ is_loaded : function (obj) { obj = this.get_node(obj); return obj && ( (obj === -1 && !this.get_container().find("> ul > li.jstree-loading").length) || ( obj !== -1 && !obj.hasClass('jstree-loading') && (obj.find('> ul > li').length || obj.hasClass('jstree-leaf')) ) ); },
+ /*
+ Function: is_loading
+ Check if a node is currently loading.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+
+ Returns:
+ true - _obj_ is currently loading
+ false - _obj_ is not currently loading
+ */
+ is_loading : function (obj) { obj = this.get_node(obj); return obj && ( (obj === -1 && this.get_container().find("> ul > li.jstree-loading").length) || (obj !== -1 && obj.hasClass("jstree-loading")) ); },
+ /*
+ Function: is_open
+ Check if a node is currently open.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+
+ Returns:
+ true - _obj_ is currently open
+ false - _obj_ is not currently open
+ */
+ is_open : function (obj) { obj = this.get_node(obj); return obj && obj !== -1 && obj.hasClass("jstree-open"); },
+ /*
+ Function: is_closed
+ Check if a node is currently closed.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+
+ Returns:
+ true - _obj_ is currently closed
+ false - _obj_ is not currently closed
+ */
+ is_closed : function (obj) { obj = this.get_node(obj); return obj && obj !== -1 && obj.hasClass("jstree-closed"); },
+ /*
+ Function: is_leaf
+ Check if a node is a leaf node (has no children).
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+
+ Returns:
+ true - _obj_ is a leaf node
+ false - _obj_ is not a leaf node
+ */
+ is_leaf : function (obj) { obj = this.get_node(obj); return obj && obj !== -1 && obj.hasClass("jstree-leaf"); },
+ /*
+ Function: load_node
+ Load the children of a node.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc. Use -1 to load the root nodes.
+ callback - a function to be executed in the tree's scope. Receives two arguments: _obj_ (the same node used to call load_node), _status_ (a boolean indicating if the node was loaded successfully.
+
+ Returns:
+ true - _obj_ is a valid node and will try loading it
+ false - _obj_ is not a valid node
+
+ Triggers:
+ <load_node>
+
+ See also:
+ <_load_node>
+
+ Event: load_node
+ This event is triggered in the *jstree* namespace when a node is loaded (succesfully or not).
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else).
+ data.rslt - *object* which contains two keys _obj_ (the loaded node) and _status_ - whether the node was loaded successfully.
+
+ Example:
+ > $("div").bind("load_node.jstree", function (e, data) { if(data.rslt.status) { data.inst.open_node(data.rslt.obj); } });
+ */
+ load_node : function (obj, callback) {
+ obj = this.get_node(obj);
+ if(!obj) { callback.call(this, obj, false); return false; }
+ // if(this.is_loading(obj)) { return true; }
+ if(obj !== -1) { obj.addClass("jstree-loading"); }
+ this._load_node(obj, $.proxy(function (status) {
+ if(obj !== -1) { obj.removeClass("jstree-loading"); }
+ this.__callback({ "obj" : obj, "status" : status });
+ if(callback) { callback.call(this, obj, status); }
+ }, this));
+ return true;
+ },
+ /*
+ Function: _load_node
+ Load the children of a node, but as opposed to <load_node> does not change any visual properties or trigger events. This function is used in <load_node> internally. The idea is for data source plugins to overwrite this function.
+ This implementation (from the *core*) only uses markup found in the tree container, and does not load async.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc. Use -1 to load the root nodes.
+ callback - a function to be executed in the tree's scope. Receives one argument: _status_ (a boolean indicating if the node was loaded successfully).
+ */
+ _load_node : function (obj, callback) {
+ // if using async - empty the node first
+ if(obj === -1) {
+ this.get_container_ul().empty().append(this.data.core.original_container_html.clone(true));
+ }
+ callback.call(null, true);
+ },
+ /*
+ Function: open_node
+ Open a node so that its children are visible. If the node is not loaded try loading it first.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+ callback - a function to be executed in the tree's scope. Receives two arguments: _obj_ (the node being opened) and _status_ (a boolean indicating if the node was opened successfully).
+ animation - the duration in miliseconds of the slideDown animation. If not supplied the jQuery default is used. Please note that on IE6 a _0_ is enforced here due to performance issues.
+
+ Triggers:
+ <open_node>, <__after_open>
+
+ Event: open_node
+ This event is triggered in the *jstree* namespace when a node is successfully opened (but if animation is used this event is triggered BEFORE the animation completes). See <__after_open>.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else).
+ data.rslt - *object* which contains a single key: _obj_ (the opened node).
+
+ Example:
+ > $("div").bind("open_node.jstree", function (e, data) {
+ > data.rslt.obj.find('> ul > .jstree-closed').each(function () {
+ > data.inst.open_node(this);
+ > }
+ > });
+
+ Event: __after_open
+ This event is triggered in the *jstree* namespace when a node is successfully opened AFTER the animation completes). See <open_node>.
+
+ Parameters:
+ data.inst - the instance
+ data.rslt - *object* which contains a single key: _obj_ (the opened node).
+
+ Example:
+ > $("div").bind("__after_open.jstree", function (e, data) {
+ > data.rslt.obj.find('> ul > .jstree-closed').each(function () {
+ > data.inst.open_node(this);
+ > }
+ > });
+ */
+ open_node : function (obj, callback, animation) {
+ obj = this.get_node(obj);
+ if(obj === -1 || !obj || !obj.length) { return false; }
+ if(!this.is_closed(obj)) { if(callback) { callback.call(this, obj, false); } return false; }
+ if(!this.is_loaded(obj)) { // TODO: is_loading?
+ this.load_node(obj, function (o, ok) {
+ return ok ? this.open_node(o, callback, animation) : callback ? callback.call(this, o, false) : false;
+ });
+ }
+ else {
+ var t = this;
+ obj
+ .children("ul").css("display","none").end()
+ .removeClass("jstree-closed").addClass("jstree-open")
+ // .children("ins").text("-").end()
+ .children("ul").stop(true, true).slideDown( ($.jstree.IS_IE6 ? 0 : animation), function () {
+ this.style.display = "";
+ t.__trigger("__after_open", { "rslt" : { "obj" : obj } });
+ });
+ if(callback) { callback.call(this, obj, true); }
+ this.__callback({ "obj" : obj });
+ }
+ },
+ /*
+ Function: close_node
+ Close a node so that its children are not visible.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+ animation - the duration in miliseconds of the slideDown animation. If not supplied the jQuery default is used. Please note that on IE6 a _0_ is enforced here due to performance issues.
+
+ Triggers:
+ <close_node>, <__after_close>
+
+ Event: close_node
+ This event is triggered in the *jstree* namespace when a node is closed (but if animation is used this event is triggered BEFORE the animation completes). See <__after_close>.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else).
+ data.rslt - *object* which contains a single key: _obj_ (the closed node).
+
+ Example:
+ > $("div").bind("close_node.jstree", function (e, data) {
+ > data.rslt.obj.children('ul').remove();
+ > });
+
+ Event: __after_close
+ This event is triggered in the *jstree* namespace when a node is closed AFTER the animation completes). See <close_node>.
+
+ Parameters:
+ data.inst - the instance
+ data.rslt - *object* which contains a single key: _obj_ (the opened node).
+
+ Example:
+ > $("div").bind("__after_close.jstree", function (e, data) {
+ > data.rslt.obj.children('ul').remove();
+ > });
+ */
+ close_node : function (obj, animation) {
+ obj = this.get_node(obj);
+ if(!obj || !obj.length || !this.is_open(obj)) { return false; }
+ var t = this;
+ obj
+ .children("ul").attr("style","display:block !important").end()
+ .removeClass("jstree-open").addClass("jstree-closed")
+ // .children("ins").text("+").end()
+ .children("ul").stop(true, true).slideUp( ($.jstree.IS_IE6 ? 0 : animation), function () {
+ this.style.display = "";
+ t.__trigger("__after_close", { "rslt" : { "obj" : obj } });
+ });
+ this.__callback({ "obj" : obj });
+ },
+ /*
+ Function: toggle_node
+ If a node is closed - open it, if it is open - close it.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+ */
+ toggle_node : function (obj) {
+ if(this.is_closed(obj)) { return this.open_node(obj); }
+ if(this.is_open(obj)) { return this.close_node(obj); }
+ },
+ /*
+ Function: open_all
+ Open all nodes from a certain node down.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc. If _-1_ is used or is omitted all nodes in the tree are opened.
+ animation - the duration of the slideDown animation when opening the nodes. If not set _0_ is enforced for performance issues.
+ original_obj - used internally to keep track of the recursion - do not set manually!
+
+ Triggers:
+ <open_all>
+
+ Event: open_all
+ This event is triggered in the *jstree* namespace when an open_all call completes.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else).
+ data.rslt - *object* which contains a single key: _obj_ (the node used in the call).
+
+ Example:
+ > $("div").bind("open_all.jstree", function (e, data) {
+ > alert('DONE');
+ > });
+ */
+ open_all : function (obj, animation, original_obj) {
+ obj = obj ? this.get_node(obj) : -1;
+ obj = !obj || obj === -1 ? this.get_container_ul() : obj;
+ original_obj = original_obj || obj;
+ var _this = this;
+ obj = this.is_closed(obj) ? obj.find('li.jstree-closed').andSelf() : obj.find('li.jstree-closed');
+ obj.each(function () {
+ _this.open_node(
+ this,
+ _this.is_loaded(this) ?
+ false :
+ function(obj) { this.open_all(obj, animation, original_obj); },
+ animation || 0
+ );
+ });
+ if(original_obj.find('li.jstree-closed').length === 0) { this.__callback({ "obj" : original_obj }); }
+ },
+ /*
+ Function: close_all
+ Close all nodes from a certain node down.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc. If _-1_ is used or is omitted all nodes in the tree are closed.
+ animation - the duration of the slideDown animation when closing the nodes. If not set _0_ is enforced for performance issues.
+
+ Triggers:
+ <close_all>
+
+ Event: close_all
+ This event is triggered in the *jstree* namespace when a close_all call completes.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else).
+ data.rslt - *object* which contains a single key: _obj_ (the node used in the call).
+
+ Example:
+ > $("div").bind("close_all.jstree", function (e, data) {
+ > alert('DONE');
+ > });
+ */
+ close_all : function (obj, animation) {
+ obj = obj ? this._get_node(obj) : -1;
+ var $obj = !obj || obj === -1 ? this.get_container_ul() : obj,
+ _this = this;
+ $obj = this.is_open($obj) ? $obj.find('li.jstree-open').andSelf() : $obj.find('li.jstree-open');
+ $obj.each(function () { _this.close_node(this, animation || 0); });
+ this.__callback({ "obj" : obj });
+ },
+ /*
+ Function: clean_node
+ This function converts inserted nodes to the required by jsTree format. It takes care of converting a simple unodreder list to the internally used markup.
+ The core calls this function automatically when new data arrives (by binding to the <load_node> event).
+ Each plugin may override this function to include its own source, but keep in mind to do it like that:
+ > clean_node : function(obj) {
+ > obj = this.__call_old();
+ > obj.each(function () {
+ > // do your stuff here
+ > });
+ > }
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc. If _-1_ is used or is omitted all nodes in the tree are cleaned.
+
+ Returns:
+ jQuery collection - the cleaned children of the original node.
+ */
+ clean_node : function (obj) {
+ // DETACH maybe inside the "load_node" function? But what about animations, etc?
+ obj = this.get_node(obj);
+ obj = !obj || obj === -1 ? this.get_container().find("li") : obj.find("li").andSelf();
+ var _this = this;
+ return obj.each(function () {
+ var t = $(this),
+ d = t.data("jstree"),
+ s = (d && d.opened) || t.hasClass("jstree-open") ? "open" : (d && d.closed) || t.children("ul").length ? "closed" : "leaf";
+ if(d && d.opened) { delete d.opened; }
+ if(d && d.closed) { delete d.closed; }
+ t.removeClass("jstree-open jstree-closed jstree-leaf jstree-last");
+ if(!t.children("a").length) {
+ // allow for text and HTML markup inside the nodes
+ t.contents().filter(function() { return this.nodeType === 3 || this.tagName !== 'UL'; }).wrapAll('<a href="#"></a>');
+ // TODO: make this faster
+ t.children('a').html(t.children('a').html().replace(/[\s\t\n]+$/,''));
+ }
+ else {
+ if(!$.trim(t.children('a').attr('href'))) { t.children('a').attr("href","#"); }
+ }
+ if(!t.children("ins.jstree-ocl").length) {
+ t.prepend("<ins class='jstree-icon jstree-ocl'>&#160;</ins>");
+ }
+ if(t.is(":last-child")) {
+ t.addClass("jstree-last");
+ }
+ switch(s) {
+ case 'leaf':
+ t.addClass('jstree-leaf');
+ break;
+ case 'closed':
+ t.addClass('jstree-open');
+ _this.close_node(t, 0);
+ break;
+ case 'open':
+ t.addClass('jstree-closed');
+ _this.open_node(t, false, 0);
+ break;
+ }
+ });
+ },
+ /*
+ Function: correct_node
+ This function corrects the open/closed/leaf state as data changes (as the user interacts with the tree).
+ The core calls this function automatically when a node is opened, deleted or moved.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc. If _-1_ is used or is omitted the root nodes are processed.
+
+ Returns:
+ jQuery collection - the processed children of the original node.
+ */
+ /* PROCESS SINGLE NODE (OR USE BOOLEAN single PARAM), CALL FROM CLEAN_NODE, LOSE THE EVENTS ABOVE */
+ correct_node : function (obj, deep) {
+ obj = this.get_node(obj);
+ if(!obj || (obj === -1 && !deep)) { return false; }
+ if(obj === -1) { obj = this.get_container().find('li'); }
+ else { obj = deep ? obj.find('li').andSelf() : obj; }
+ obj.each(function () {
+ var obj = $(this);
+ switch(!0) {
+ case obj.hasClass("jstree-open") && !obj.find("> ul > li").length:
+ obj.removeClass("jstree-open").addClass("jstree-leaf").children("ul").remove(); // children("ins").html("&#160;").end()
+ break;
+ case obj.hasClass("jstree-leaf") && !!obj.find("> ul > li").length:
+ obj.removeClass("jstree-leaf").addClass("jstree-closed"); //.children("ins").html("+");
+ break;
+ }
+ obj[obj.is(":last-child") ? 'addClass' : 'removeClass']("jstree-last");
+ });
+ return obj;
+ },
+ /*
+ Function: scroll_to_node
+ This function scrolls the container to the desired node (if needed).
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+ */
+ scroll_to_node : function (obj) {
+ var c = this.get_container()[0], t;
+ if(c.scrollHeight > c.offsetHeight) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length || !obj.is(":visible")) { return; }
+ t = obj.offset().top - this.get_container().offset().top;
+ if(t < 0) {
+ c.scrollTop = c.scrollTop + t - 1;
+ }
+ if(t + this.data.core.li_height + (c.scrollWidth > c.offsetWidth ? $.jstree.SCROLLBAR_WIDTH : 0) > c.offsetHeight) {
+ c.scrollTop = c.scrollTop + (t - c.offsetHeight + this.data.core.li_height + 1 + (c.scrollWidth > c.offsetWidth ? $.jstree.SCROLLBAR_WIDTH : 0));
+ }
+ }
+ },
+ /*
+ Function: get_state
+ This function returns the current state of the tree (as collected from all active plugins).
+ Plugin authors: pay special attention to the way this function is extended for new plugins. In your plugin code write:
+ > get_state : function () {
+ > var state = this.__call_old();
+ > state.your-plugin-name = <some-value-you-collect>;
+ > return state;
+ > }
+
+ Returns:
+ object - the current state of the instance
+ */
+ get_state : function () { // TODO: scroll position, theme
+ var state = { 'open' : [], 'scroll' : { 'left' : this.get_container().scrollLeft(), 'top' : this.get_container().scrollTop() } };
+ this.get_container_ul().find('.jstree-open').each(function () { if(this.id) { state.open.push(this.id); } });
+ return state;
+ },
+ /*
+ Function: set_state
+ This function returns sets the state of the tree.
+ Plugin authors: pay special attention to the way this function is extended for new plugins. In your plugin code write:
+ > set_state : function (state, callback) {
+ > if(this.__call_old()) {
+ > if(state.your-plugin-name) {
+ >
+ > // restore using `state.your-plugin-name`
+ > // if you need some async activity so that you return to this bit of code
+ > // do not delete state.your-plugin-name and return false (see core's function for example)
+ >
+ > delete state.your-plugin-name;
+ > this.set_state(state, callback);
+ > return false;
+ > }
+ > return true;
+ > }
+ > return false;
+ > }
+
+ Parameters:
+ state - *object* the state to restore to
+ callback - *function* this will be executed in the instance's scope once restoring is done
+
+ Returns:
+ boolean - the return value is used to determine the phase of restoration
+
+ Triggers:
+ <set_state>
+
+ Event: set_state
+ This event is triggered in the *jstree* namespace when a set_state call completes.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else)
+ */
+ set_state : function (state, callback) {
+ if(state) {
+ if($.isArray(state.open)) {
+ var res = true,
+ t = this;
+ this.close_all();
+ $.each(state.open.concat([]), function (i, v) {
+ v = document.getElementById(v);
+ if(v) {
+ if(t.is_loaded(v)) {
+ if(t.is_closed(v)) {
+ t.open_node(v, false, 0);
+ }
+ $.vakata.array_remove(state.open, i);
+ }
+ else {
+ t.open_node(v, $.proxy(function () { this.set_state(state); }, t), 0);
+ // there will be some async activity - so wait for it
+ res = false;
+ }
+ }
+ });
+ if(res) {
+ delete state.open;
+ this.set_state(state, callback);
+ }
+ return false;
+ }
+ if(state.scroll) {
+ if(state.scroll && typeof state.scroll.left !== 'undefined') {
+ this.get_container().scrollLeft(state.scroll.left);
+ }
+ if(state.scroll && typeof state.scroll.top !== 'undefined') {
+ this.get_container().scrollTop(state.scroll.top);
+ }
+ delete state.scroll;
+ delete state.open;
+ this.set_state(state, callback);
+ return false;
+ }
+ if($.isEmptyObject(state)) {
+ if(callback) { callback.call(this); }
+ this.__callback();
+ return false;
+ }
+ return true;
+ }
+ return false;
+ },
+ /*
+ Function: refresh
+ This function saves the current state, reloads the complete tree and returns it to the saved state.
+
+ Triggers:
+ <refresh>
+
+ Event: refresh
+ This event is triggered in the *jstree* namespace when a refresh call completes.
+
+ Parameters:
+ data.inst - the instance
+ */
+ refresh : function () {
+ this.data.core.state = this.get_state();
+ this.load_node(-1, function (o, s) {
+ if(s) {
+ this.set_state($.extend(true, {}, this.data.core.state), function () { this.__trigger('refresh'); });
+ }
+ this.data.core.state = null;
+ });
+ },
+ /*
+ Function: get_text
+ This function returns the title of the node.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+ remove_html - *boolean* set to _true_ to return plain text instead of HTML
+
+ Returns:
+ string - the title of the node, specified by _obj_
+ */
+ get_text : function (obj, remove_html) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ obj = obj.children("a:eq(0)").clone();
+ obj.children(".jstree-icon").remove();
+ return obj[ remove_html ? 'text' : 'html' ]();
+ },
+ /*
+ Function: set_text
+ This function sets the title of the node. This is a low-level function, you'd be better off using <rename>.
+
+ Parameters:
+ obj - *mixed* this is used as a jquery selector - can be jQuery object, DOM node, string, etc.
+ val - *string* the new title of the node (can be HTMl too)
+
+ Returns:
+ boolean - was the rename successfull
+
+ Triggers:
+ <set_text>
+
+ Event: set_text
+ This event is triggered in the *jstree* namespace when a set_text call completes.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else)
+ data.rslt - *object* which contains a two keys: _obj_ (the node) and _val_ (the new title).
+
+ Example:
+ > $("div").bind("set_text.jstree", function (e, data) {
+ > alert("Renamed to: " + data.rslt.val);
+ > });
+ */
+ set_text : function (obj, val) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ obj = obj.children("a:eq(0)");
+ var tmp = obj.children("INS").clone();
+ obj.html(val).prepend(tmp);
+ this.__callback({ "obj" : obj, "text" : val });
+ return true;
+ },
+ /*
+ Function: parse_json
+ This function returns a jQuery node after parsing a JSON object (a LI node for single elements or an UL node for multiple). This function will use the default title from <jstree.config.core.strings> if none is specified.
+
+ Parameters:
+ node - *mixed* the input to parse
+ > // can be a string
+ > "The title of the parsed node"
+ > // array of strings
+ > [ "Node 1", "Node 2" ]
+ > // an object
+ > { "title" : "The title of the parsed node" }
+ > // you can manipulate the output
+ > { "title" : "The title of the parsed node", "li_attr" : { "id" : "id_for_li" }, "a_attr" : { "href" : "http://jstree.com" } }
+ > // you can supply metadata, which you can later access using $(the_li_node).data()
+ > { "title" : "The title of the parsed node", "data" : { <some-values-here> } }
+ > // you can supply children (they can be objects too)
+ > { "title" : "The title of the parsed node", "children" : [ "Node 1", { "title" : "Node 2" } ] }
+
+ Returns:
+ jQuery - the LI (or UL) node which was produced from the JSON
+ */
+ parse_json : function (node) {
+ var li, a, ul, t;
+ if($.isArray(node)) {
+ ul = $("<ul />");
+ t = this;
+ $.each(node, function (i, v) {
+ ul.append(t.parse_json(v));
+ });
+ return ul;
+ }
+ if(typeof node === "undefined") { node = {}; }
+ if(typeof node === "string") { node = { "title" : node }; }
+ if(!node.li_attr) { node.li_attr = {}; }
+ if(!node.a_attr) { node.a_attr = {}; }
+ if(!node.a_attr.href) { node.a_attr.href = '#'; }
+ if(!node.title) { node.title = this._get_string("New node"); }
+
+ li = $("<li />").attr(node.li_attr);
+ a = $("<a />").attr(node.a_attr).html(node.title);
+ ul = $("<ul />");
+ if(node.data && !$.isEmptyObject(node.data)) { li.data(node.data); }
+ if(
+ node.children === true ||
+ $.isArray(node.children) ||
+ (li.data('jstree') && $.isArray(li.data('jstree').children))
+ ) {
+ if(!li.data('jstree')) {
+ li.data('jstree', {});
+ }
+ li.data('jstree').closed = true;
+ }
+ li.append(a);
+ if($.isArray(node.children)) {
+ $.each(node.children, $.proxy(function (i, n) {
+ ul.append(this.parse_json(n));
+ }, this));
+ li.append(ul);
+ }
+ return li;
+ },
+ /*
+ Function: get_json
+ This function returns the whole tree (or a single node) in JSON format.
+ Each plugin may override this function to include its own source, but keep in mind to do it like that:
+ > get_json : function(obj, is_callback) {
+ > var r = this.__call_old();
+ > if(is_callback) {
+ > if(<some-condition>) { r.data.jstree.<some-key> = <some-value-this-plugin-will-process>; }
+ > }
+ > return r;
+ > }
+
+ Parameters:
+ obj - *mixed* the input to parse
+ is_callback - do not modify this, jstree uses this parameter to keep track of the recursion
+
+ Returns:
+ Array - an array consisting of objects (one for each node)
+ */
+ get_json : function (obj, is_callback) {
+ obj = typeof obj !== 'undefined' ? this.get_node(obj) : false;
+ if(!is_callback) {
+ if(!obj || obj === -1) { obj = this.get_container_ul().children("li"); }
+ }
+ var r, t, li_attr = {}, a_attr = {}, tmp = {};
+ if(!obj || !obj.length) { return false; }
+ if(obj.length > 1 || !is_callback) {
+ r = [];
+ t = this;
+ obj.each(function () {
+ r.push(t.get_json($(this), true));
+ });
+ return r;
+ }
+ tmp = $.vakata.attributes(obj, true);
+ $.each(tmp, function (i, v) {
+ if(i == 'id') { li_attr[i] = v; return true; }
+ v = $.trim(v.replace(/\bjstree[^ ]*/ig,'').replace(/\s+$/ig," "));
+ if(v.length) { li_attr[i] = v; }
+ });
+ tmp = $.vakata.attributes(obj.children('a'), true);
+ $.each(tmp, function (i, v) {
+ if(i == 'id') { a_attr[i] = v; return true; }
+ v = $.trim(v.replace(/\bjstree[^ ]*/ig,'').replace(/\s+$/ig," "));
+ if(v.length) { a_attr[i] = v; }
+ });
+ r = {
+ 'title' : this.get_text(obj),
+ 'data' : $.extend(true, {}, obj.data() || {}),
+ 'children' : false,
+ 'li_attr' : li_attr,
+ 'a_attr' : a_attr
+ };
+
+ if(!r.data.jstree) { r.data.jstree = {}; }
+ if(this.is_open(obj)) { r.data.jstree.opened = true; }
+ if(this.is_closed(obj)) { r.data.jstree.closed = true; }
+
+ obj = obj.find('> ul > li');
+ if(obj.length) {
+ r.children = [];
+ t = this;
+ obj.each(function () {
+ r.children.push(t.get_json($(this), true));
+ });
+ }
+ return r;
+ },
+ /*
+ Function: create_node
+ This function creates a new node.
+
+ Parameters:
+ parent - *mixed* the parent for the newly created node. This is used as a jquery selector, can be jQuery object, DOM node, string, etc. Use -1 to create a new root node.
+ node - *mixed* the input to parse, check <parse_json> for description
+ position - *mixed* where to create the new node. Can be one of "before", "after", "first", "last", "inside" or a numerical index.
+ callback - optional function to be executed once the node is created
+ is_loaded - used internally when a node needs to be loaded - do not pass this
+
+ Returns:
+ jQuery - the LI node which was produced from the JSON (may return _undefined_ if the parent node is not yet loaded, but will create the node)
+
+ Triggers:
+ <create_node>
+
+ Event: create_node
+ This event is triggered in the *jstree* namespace when a new node is created.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else)
+ data.rslt - *object* which contains a three keys: _obj_ (the node), _parent_ (the parent) and _position_ which is the numerical index.
+
+ Example:
+ > $("div").bind("create_node.jstree", function (e, data) {
+ > alert("Created `" + data.inst.get_text(data.rslt.obj) + "` inside `" + (data.rslt.parent === -1 ? 'the main container' : data.inst.get_text(data.rslt.parent)) + "` at index " + data.rslt.position);
+ > });
+ */
+ create_node : function (par, node, pos, callback, is_loaded) {
+ par = this.get_node(par);
+ pos = typeof pos === "undefined" ? "last" : pos;
+
+ if(par !== -1 && !par.length) { return false; }
+ if(!pos.match(/^(before|after)$/) && !is_loaded && !this.is_loaded(par)) {
+ return this.load_node(par, function () { this.create_node(par, node, pos, callback, true); });
+ }
+
+ var li = this.parse_json(node),
+ tmp = par === -1 ? this.get_container() : par;
+
+ if(par === -1) {
+ if(pos === "before") { pos = "first"; }
+ if(pos === "after") { pos = "last"; }
+ }
+ switch(pos) {
+ case "before":
+ pos = par.index();
+ par = this.get_parent(par);
+ break;
+ case "after" :
+ pos = par.index() + 1;
+ par = this.get_parent(par);
+ break;
+ case "inside":
+ case "first":
+ pos = 0;
+ break;
+ case "last":
+ pos = tmp.children('ul').children('li').length;
+ break;
+ default:
+ if(!pos) { pos = 0; }
+ break;
+ }
+ if(!this.check("create_node", li, par, pos)) { return false; }
+
+ tmp = par === -1 ? this.get_container() : par;
+ if(!tmp.children("ul").length) { tmp.append("<ul />"); }
+ if(tmp.children("ul").children("li").eq(pos).length) {
+ tmp.children("ul").children("li").eq(pos).before(li);
+ }
+ else {
+ tmp.children("ul").append(li);
+ }
+ this.correct_node(par, true);
+ if(callback) { callback.call(this, li); }
+ this.__callback({ "obj" : li, "parent" : par, "position" : li.index() });
+ return li;
+ },
+ /*
+ Function: rename_node
+ This function renames a new node.
+
+ Parameters:
+ obj - *mixed* the node to rename. This is used as a jquery selector, can be jQuery object, DOM node, string, etc.
+ val - *string* the new title
+
+ Triggers:
+ <rename_node>
+
+ Event: rename_node
+ This event is triggered in the *jstree* namespace when a node is renamed.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else)
+ data.rslt - *object* which contains a three keys: _obj_ (the node), _title_ (the new title), _old_ (the old title)
+
+ Example:
+ > $("div").bind("rename_node.jstree", function (e, data) {
+ > alert("Node rename from `" + data.rslt.old + "` to `" + data.rslt.title "`");
+ > });
+ */
+ rename_node : function (obj, val) {
+ obj = this.get_node(obj);
+ var old = this.get_text(obj);
+ if(!this.check("rename_node", obj, this.get_parent(obj), val)) { return false; }
+ if(obj && obj.length) {
+ this.set_text(obj, val); // .apply(this, Array.prototype.slice.call(arguments))
+ this.__callback({ "obj" : obj, "title" : val, "old" : old });
+ }
+ },
+ /*
+ Function: delete_node
+ This function deletes a node.
+
+ Parameters:
+ obj - *mixed* the node to remove. This is used as a jquery selector, can be jQuery object, DOM node, string, etc.
+
+ Returns:
+ mixed - the removed node on success, _false_ on failure
+
+ Triggers:
+ <delete_node>
+
+ Event: delete_node
+ This event is triggered in the *jstree* namespace when a node is deleted.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else)
+ data.rslt - *object* which contains a three keys: _obj_ (the removed node), _prev_ (the previous sibling of the removed node), _parent_ (the parent of the removed node)
+
+ Example:
+ > $("div").bind("delete_node.jstree", function (e, data) {
+ > alert("Node deleted!");
+ > });
+ */
+ delete_node : function (obj) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ var par = this.get_parent(obj),
+ pre = this.get_prev(obj);
+ if(!this.check("delete_node", obj, par, obj.index())) { return false; }
+ obj = obj.detach();
+ this.correct_node(par);
+ this.correct_node(pre);
+ this.__callback({ "obj" : obj, "prev" : pre, "parent" : par });
+ return obj;
+ },
+ /*
+ Function: check
+ This function checks if a structure modification is valid.
+
+ Parameters:
+ chk - *string* what are we checking (copy_node, move_node, rename_node, create_node, delete_node)
+ obj - *mixed* the node.
+ par - *mixed* the parent (if dealing with a move or copy - the new parent).
+ pos - *mixed* the index among the parent's children (or the new name if dealing with a rename)
+ is_copy - *boolean* is this a copy or a move call
+
+ Returns:
+ boolean - _true_ if the move is valid, _false_ otherwise
+ */
+ check : function (chk, obj, par, pos) {
+ switch(chk) {
+ case "create_node":
+ break;
+ case "rename_node":
+ break;
+ case "move_node":
+ var tmp = par === -1 ? this.get_container() : par;
+ tmp = tmp.children('ul').children('li');
+ if(tmp.length && tmp.index(obj) !== -1 && (pos === obj.index() || pos === obj.index() + 1)) {
+ return false;
+ }
+ if(par !== -1 && par.parentsUntil('.jstree', 'li').andSelf().index(obj) !== -1) {
+ return false;
+ }
+ break;
+ case "copy_node":
+ break;
+ case "delete_node":
+ break;
+ }
+ return true;
+ },
+ /*
+ Function: move_node
+ This function moves a node.
+
+ Parameters:
+ obj - *mixed* the node to move. This is used as a jquery selector, can be jQuery object, DOM node, string, etc.
+ parent - *mixed* the new parent. This is used as a jquery selector, can be jQuery object, DOM node, string, etc. Use -1 to promote to a root node.
+ position - *mixed* where to create the new node. Can be one of "before", "after", "first", "last", "inside" or a numerical index.
+ callback - optional function to be executed once the node is moved
+ is_loaded - used internally when a node needs to be loaded - do not pass this
+
+ Returns:
+ boolean - indicating if the move was successfull (may return _undefined_ if the parent node is not yet loaded, but will move the node)
+
+
+ Triggers:
+ <move_node>
+
+ Event: move_node
+ This event is triggered in the *jstree* namespace when a node is moved.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else)
+ data.rslt - *object* which contains a five keys: _obj_ (the node), _parent_ (the new parent) and _position_ which is the numerical index, _old_parent_ (the old parent) and is_multi (a boolean indicating if the node is coming from another tree instance)
+
+ Example:
+ > $("div").bind("move_node.jstree", function (e, data) {
+ > alert("Moved `" + data.inst.get_text(data.rslt.obj) + "` inside `" + (data.rslt.parent === -1 ? 'the main container' : data.inst.get_text(data.rslt.parent)) + "` at index " + data.rslt.position);
+ > });
+ */
+ move_node : function (obj, par, pos, callback, is_loaded) {
+ obj = this.get_node(obj);
+ par = this.get_node(par);
+ pos = typeof pos === "undefined" ? 0 : pos;
+
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ if(par !== -1 && !par.length) { return false; }
+ if(!pos.toString().match(/^(before|after)$/) && !is_loaded && !this.is_loaded(par)) {
+ return this.load_node(par, function () { this.move_node(obj, par, pos, callback, true); });
+ }
+
+ var old_par = this.get_parent(obj),
+ new_par = (!pos.toString().match(/^(before|after)$/) || par === -1) ? par : this.get_parent(par),
+ old_ins = $.jstree._reference(obj),
+ new_ins = par === -1 ? this : $.jstree._reference(par),
+ is_multi = (old_ins.get_index() !== new_ins.get_index());
+ if(new_par === -1) {
+ par = new_ins.get_container();
+ if(pos === "before") { pos = "first"; }
+ if(pos === "after") { pos = "last"; }
+ }
+ switch(pos) {
+ case "before":
+ pos = par.index();
+ break;
+ case "after" :
+ pos = par.index() + 1;
+ break;
+ case "inside":
+ case "first":
+ pos = 0;
+ break;
+ case "last":
+ pos = par.children('ul').children('li').length;
+ break;
+ default:
+ if(!pos) { pos = 0; }
+ break;
+ }
+ if(!this.check("move_node", obj, new_par, pos)) { return false; }
+
+ if(!par.children("ul").length) { par.append("<ul />"); }
+ if(par.children("ul").children("li").eq(pos).length) {
+ par.children("ul").children("li").eq(pos).before(obj);
+ }
+ else {
+ par.children("ul").append(obj);
+ }
+
+ if(is_multi) { // if multitree - clean the node recursively - remove all icons, and call deep clean_node
+ obj.find('.jstree-icon, .jstree-ocl').remove();
+ this.clean_node(obj);
+ }
+ old_ins.correct_node(old_par, true);
+ new_ins.correct_node(new_par, true);
+ if(callback) { callback.call(this, obj, new_par, obj.index()); }
+ this.__callback({ "obj" : obj, "parent" : new_par, "position" : obj.index(), "old_parent" : old_par, "is_multi" : is_multi, 'old_instance' : old_ins, 'new_instance' : new_ins });
+ return true;
+ },
+ /*
+ Function: copy_node
+ This function copies a node.
+
+ Parameters:
+ obj - *mixed* the node to copy. This is used as a jquery selector, can be jQuery object, DOM node, string, etc.
+ parent - *mixed* the new parent. This is used as a jquery selector, can be jQuery object, DOM node, string, etc. Use -1 to promote to a root node.
+ position - *mixed* where to create the new node. Can be one of "before", "after", "first", "last", "inside" or a numerical index.
+ callback - optional function to be executed once the node is moved
+ is_loaded - used internally when a node needs to be loaded - do not pass this
+
+ Returns:
+ boolean - indicating if the move was successfull (may return _undefined_ if the parent node is not yet loaded, but will move the node)
+
+
+ Triggers:
+ <copy_node>
+
+ Event: copy_node
+ This event is triggered in the *jstree* namespace when a node is copied.
+
+ Parameters:
+ data.inst - the instance
+ data.args - *array* the arguments passed to the function
+ data.plugin - *string* the function's plugin (here it will be _"core"_ but if the function is extended it may be something else)
+ data.rslt - *object* which contains a five keys: _obj_ (the node), _parent_ (the new parent) and _position_ which is the numerical index, _original_ (the original object), is_multi (a boolean indicating if the node is coming from another tree instance, _old_instance_ (the source instance) and _new_instance_ (the receiving instance))
+
+ Example:
+ > $("div").bind("copy_node.jstree", function (e, data) {
+ > alert("Copied `" + data.inst.get_text(data.rslt.original) + "` inside `" + (data.rslt.parent === -1 ? 'the main container' : data.inst.get_text(data.rslt.parent)) + "` at index " + data.rslt.position);
+ > });
+ */
+ copy_node : function (obj, par, pos, callback, is_loaded) {
+ obj = this.get_node(obj);
+ par = this.get_node(par);
+ pos = typeof pos === "undefined" ? "last" : pos;
+
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ if(par !== -1 && !par.length) { return false; }
+ if(!pos.toString().match(/^(before|after)$/) && !is_loaded && !this.is_loaded(par)) {
+ return this.load_node(par, function () { this.copy_node(obj, par, pos, callback, true); });
+ }
+ var org_obj = obj,
+ old_par = this.get_parent(obj),
+ new_par = (!pos.toString().match(/^(before|after)$/) || par === -1) ? par : this.get_parent(par),
+ old_ins = $.jstree._reference(obj),
+ new_ins = par === -1 ? this : $.jstree._reference(par),
+ is_multi = (old_ins.get_index() !== new_ins.get_index());
+
+ obj = obj.clone(true);
+ obj.find("*[id]").andSelf().each(function () {
+ if(this.id) { this.id = "copy_" + this.id; }
+ });
+ if(new_par === -1) {
+ par = new_ins.get_container();
+ if(pos === "before") { pos = "first"; }
+ if(pos === "after") { pos = "last"; }
+ }
+ switch(pos) {
+ case "before":
+ pos = par.index();
+ break;
+ case "after" :
+ pos = par.index() + 1;
+ break;
+ case "inside":
+ case "first":
+ pos = 0;
+ break;
+ case "last":
+ pos = par.children('ul').children('li').length;
+ break;
+ default:
+ if(!pos) { pos = 0; }
+ break;
+ }
+
+ if(!this.check("copy_node", org_obj, new_par, pos)) { return false; }
+
+ if(!par.children("ul").length) { par.append("<ul />"); }
+ if(par.children("ul").children("li").eq(pos).length) {
+ par.children("ul").children("li").eq(pos).before(obj);
+ }
+ else {
+ par.children("ul").append(obj);
+ }
+ if(is_multi) { // if multitree - clean the node recursively - remove all icons, and call deep clean_node
+ obj.find('.jstree-icon, .jstree-ocl').remove();
+ }
+ new_ins.clean_node(obj); // always clean so that selected states, etc. are removed
+ new_ins.correct_node(new_par, true); // no need to correct the old parent, as nothing has changed there
+ if(callback) { callback.call(this, obj, new_par, obj.index(), org_obj); }
+ this.__callback({ "obj" : obj, "parent" : new_par, "old_parent" : old_par, "position" : obj.index(), "original" : org_obj, "is_multi" : is_multi, 'old_instance' : old_ins, 'new_instance' : new_ins });
+ return true;
+ },
+
+ cut : function (obj) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ ccp_node = obj;
+ ccp_mode = 'move_node';
+ this.__callback({ "obj" : obj });
+ },
+ copy : function (obj) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ ccp_node = obj;
+ ccp_mode = 'copy_node';
+ this.__callback({ "obj" : obj });
+ },
+ can_paste : function () {
+ return ccp_mode !== false && ccp_node !== false;
+ },
+ paste : function (obj) {
+ obj = this.get_node(obj);
+ if(!obj || !ccp_mode || !ccp_mode.match(/^(copy_node|move_node)$/) || !ccp_node) { return false; }
+ this[ccp_mode](ccp_node, obj);
+ this.__callback({ "obj" : obj, "nodes" : ccp_node, "mode" : ccp_mode });
+ ccp_node = false;
+ ccp_mode = false;
+ },
+
+ edit : function (obj, default_text) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ var rtl = this.data.core.rtl,
+ w = this.get_container().width(),
+ a = obj.children('a:eq(0)'),
+ oi = obj.children("ins"),
+ ai = a.children("ins"),
+ w1 = oi.width() * oi.length,
+ w2 = ai.width() * ai.length,
+ t = typeof default_text === 'string' ? default_text : this.get_text(obj),
+ h1 = $("<div />", { css : { "position" : "absolute", "top" : "-200px", "left" : (rtl ? "0px" : "-1000px"), "visibility" : "hidden" } }).appendTo("body"),
+ h2 = obj.css("position","relative").append(
+ $("<input />", {
+ "value" : t,
+ "class" : "jstree-rename-input",
+ // "size" : t.length,
+ "css" : {
+ "padding" : "0",
+ "border" : "1px solid silver",
+ "position" : "absolute",
+ "left" : (rtl ? "auto" : (w1 + w2 + 4) + "px"),
+ "right" : (rtl ? (w1 + w2 + 4) + "px" : "auto"),
+ "top" : "0px",
+ "height" : (this.data.core.li_height - 2) + "px",
+ "lineHeight" : (this.data.core.li_height - 2) + "px",
+ "width" : "150px" // will be set a bit further down
+ },
+ "blur" : $.proxy(function () {
+ var i = obj.children(".jstree-rename-input"),
+ v = i.val();
+ if(v === "") { v = t; }
+ h1.remove();
+ i.remove();
+ this.rename_node(obj, v);
+ obj.css("position", "");
+ }, this),
+ "keyup" : function (event) {
+ var key = event.keyCode || event.which;
+ if(key == 27) { this.value = t; this.blur(); return; }
+ else if(key == 13) { this.blur(); return; }
+ else { h2.width(Math.min(h1.text("pW" + this.value).width(),w)); }
+ },
+ "keypress" : function(event) {
+ var key = event.keyCode || event.which;
+ if(key == 13) { return false; }
+ }
+ })
+ ).children(".jstree-rename-input"),
+ fn = {
+ fontFamily : a.css('fontFamily') || '',
+ fontSize : a.css('fontSize') || '',
+ fontWeight : a.css('fontWeight') || '',
+ fontStyle : a.css('fontStyle') || '',
+ fontStretch : a.css('fontStretch') || '',
+ fontVariant : a.css('fontVariant') || '',
+ letterSpacing : a.css('letterSpacing') || '',
+ wordSpacing : a.css('wordSpacing') || ''
+ };
+ this.set_text(obj, "");
+ h1.css(fn);
+ h2.css(fn).width(Math.min(h1.text("pW" + h2[0].value).width(),w))[0].select();
+ }
+ }
+ });
+
+ // add core CSS
+ $(function() {
+ var css_string = '' +
+ '.jstree ul, .jstree li { display:block; margin:0 0 0 0; padding:0 0 0 0; list-style-type:none; } ' +
+ '.jstree li { display:block; min-height:18px; line-height:18px; white-space:nowrap; margin-left:18px; min-width:18px; } ' +
+ '.jstree-rtl li { margin-left:0; margin-right:18px; } ' +
+ '.jstree > ul > li { margin-left:0px; } ' +
+ '.jstree-rtl > ul > li { margin-right:0px; } ' +
+ '.jstree .jstree-icon { display:inline-block; text-decoration:none; margin:0; padding:0; vertical-align:top; } ' +
+ '.jstree .jstree-ocl { width:18px; height:18px; text-align:center; line-height:18px; cursor:default; vertical-align:top; } ' +
+ '.jstree a { display:inline-block; line-height:16px; height:16px; color:black; white-space:nowrap; padding:1px 2px; margin:0; } ' +
+ '.jstree a:focus { outline: none; } ' +
+ 'li.jstree-open > ul { display:block; } ' +
+ 'li.jstree-closed > ul { display:none; } ';
+ // Correct IE 6 (does not support the > CSS selector)
+ if($.jstree.IS_IE6) {
+ try { document.execCommand("BackgroundImageCache", false, true); } catch (err) { } // prevents flickers
+ css_string += '' +
+ '.jstree li { height:18px; margin-left:0; margin-right:0; } ' +
+ '.jstree li li { margin-left:18px; } ' +
+ '.jstree-rtl li li { margin-left:0px; margin-right:18px; } ' +
+ 'li.jstree-open ul { display:block; } ' +
+ 'li.jstree-closed ul { display:none !important; } ' +
+ '.jstree li a { display:inline; border-width:0 !important; padding:0px 2px !important; } ';
+ }
+ // Correct IE 7 (shifts anchor nodes onhover)
+ if($.jstree.IS_IE7) {
+ css_string += '.jstree li a { border-width:0 !important; padding:0px 2px !important; } ';
+ }
+ // Correct ff2 lack of display:inline-block
+ if($.jstree.IS_FF2) {
+ css_string += '' +
+ '.jstree .jstree-icon { display:-moz-inline-box; } ' +
+ '.jstree li { line-height:12px; } ' + // WHY??
+ '.jstree a { display:-moz-inline-box; } ';
+ /* за темите
+ '.jstree .jstree-no-icons .jstree-checkbox { display:-moz-inline-stack !important; } ';
+ */
+ }
+ // the default stylesheet
+ $.vakata.css.add_sheet({ str : css_string, title : "jstree" });
+ });
+})(jQuery);
+//*/
+
+})(); \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.dnd.js b/idrop-web/web-app/js/jstree.dnd.js
new file mode 100644
index 0000000..01fbff4
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.dnd.js
@@ -0,0 +1,162 @@
+/* File: jstree.dnd.js
+Enables drag'n'drop.
+*/
+/* Group: jstree drag'n'drop plugin */
+
+(function ($) {
+ $.jstree.plugin("dnd", {
+ __construct : function () {
+ this.get_container()
+ .delegate('a', 'mousedown', $.proxy(function (e) {
+ var obj = this.get_node(e.target);
+ if(obj && obj !== -1 && obj.length && e.which === 1) { // TODO: think about e.which
+ this.get_container().trigger('mousedown.jstree');
+ return $.vakata.dnd.start(e, { 'jstree' : true, 'origin' : this, 'obj' : obj }, '<div id="jstree-dnd" class="' + (this.data.themes ? 'jstree-' + this.get_theme() : '') + '"><ins class="jstree-icon jstree-er">&#160;</ins>' + this.get_text(e.currentTarget, true) + '<ins class="jstree-copy" style="display:none;">+</ins></div>');
+ }
+ }, this));
+ },
+ // TODO: is check_timeout or is it OK as is?
+ // TODO: drag foreign items / drop foreign items (pretty easy with dnd events, but need to move marker placement in a function)
+ defaults : {
+ copy_modifier : 'ctrl',
+ open_timeout : 500
+ }
+ });
+
+ $(function() {
+ // bind only once for all instances
+ var lastmv = false,
+ opento = false,
+ marker = $('<div id="jstree-marker">&#160;</div>').hide().appendTo('body');
+
+ $(document)
+ .bind('dnd_start.vakata', function (e, data) {
+ lastmv = false;
+ })
+ .bind('dnd_move.vakata', function (e, data) {
+ if(opento) { clearTimeout(opento); }
+ if(!data.data.jstree) { return; }
+
+ // if we are hovering the marker image do nothing (can happen on "inside" drags)
+ if(data.event.target.id && data.event.target.id === 'jstree-marker') {
+ return;
+ }
+
+ var ins = $.jstree._reference(data.event.target),
+ ref = false,
+ off = false,
+ rel = false,
+ l, t, h, p, i, o;
+ // if we are over an instance
+ if(ins && ins.data && ins.data.dnd) {
+ marker.attr('class', (ins.data.themes ? 'jstree-' + ins.get_theme() : ''));
+ data.helper
+ .children().attr('class', (ins.data.themes ? 'jstree-' + ins.get_theme() : ''))
+ .find('.jstree-copy:eq(0)')[ data.event[data.data.origin.get_settings().dnd.copy_modifier + "Key"] ? 'show' : 'hide' ]();
+
+
+ // if are hovering the container itself add a new root node
+ if(data.event.target === ins.get_container()[0] || data.event.target === ins.get_container_ul()[0]) {
+ if(ins.check( (data.event[data.data.origin.get_settings().dnd.copy_modifier + "Key"] ? "copy_node" : "move_node"), data.data.obj, -1, 'last')) {
+ lastmv = { 'ins' : ins, 'par' : -1, 'pos' : 'last' };
+ marker.hide();
+ data.helper.find('.jstree-icon:eq(0)').removeClass('jstree-er').addClass('jstree-ok');
+ return;
+ }
+ }
+ else {
+ // if we are hovering a tree node
+ ref = $(data.event.target).closest('a');
+ if(ref && ref.length && ref.parent().is('.jstree-closed, .jstree-open, .jstree-leaf')) {
+ off = ref.offset();
+ rel = data.event.pageY - off.top;
+ h = ref.height();
+ if(rel < h / 3) {
+ o = ['b', 'i', 'a'];
+ }
+ else if(rel > h - h / 3) {
+ o = ['a', 'i', 'b'];
+ }
+ else {
+ o = rel > h / 2 ? ['i', 'a', 'b'] : ['i', 'b', 'a'];
+ }
+ $.each(o, function (j, v) {
+ switch(v) {
+ case 'b':
+ l = off.left - 6;
+ t = off.top - 5;
+ p = ins.get_parent(ref);
+ i = ref.parent().index();
+ break;
+ case 'i':
+ l = off.left - 2;
+ t = off.top - 5 + h / 2 + 1;
+ p = ref.parent();
+ i = 0;
+ break;
+ case 'a':
+ l = off.left - 6;
+ t = off.top - 5 + h + 2;
+ p = ins.get_parent(ref);
+ i = ref.parent().index() + 1;
+ break;
+ }
+ /*
+ // TODO: moving inside, but the node is not yet loaded?
+ // the check will work anyway, as when moving the node will be loaded first and checked again
+ if(v === 'i' && !ins.is_loaded(p)) { }
+ */
+ if(ins.check((data.event[data.data.origin.get_settings().dnd.copy_modifier + "Key"] ? "copy_node" : "move_node"),data.data.obj, p, i)) {
+ if(v === 'i' && ref.parent().is('.jstree-closed') && ins.get_settings(true).dnd.open_timeout) {
+ opento = setTimeout((function (x, z) { return function () { x.open_node(z); }; })(ins, ref), ins.get_settings(true).dnd.open_timeout);
+ }
+ lastmv = { 'ins' : ins, 'par' : p, 'pos' : i };
+ marker.css({ 'left' : l + 'px', 'top' : t + 'px' }).show();
+ data.helper.find('.jstree-icon:eq(0)').removeClass('jstree-er').addClass('jstree-ok');
+ o = true;
+ return false;
+ }
+ });
+ if(o === true) { return; }
+ }
+ }
+ }
+ lastmv = false;
+ data.helper.find('.jstree-icon').removeClass('jstree-ok').addClass('jstree-er');
+ marker.hide();
+ })
+ .bind('dnd_scroll.vakata', function (e, data) {
+ if(!data.data.jstree) { return; }
+ marker.hide();
+ lastmv = false;
+ data.helper.find('.jstree-icon:eq(0)').removeClass('jstree-ok').addClass('jstree-er');
+ })
+ .bind('dnd_stop.vakata', function (e, data) {
+ if(opento) { clearTimeout(opento); }
+ if(!data.data.jstree) { return; }
+ marker.hide();
+ if(lastmv) {
+ lastmv.ins[ data.event[data.data.origin.get_settings().dnd.copy_modifier + "Key"] ? 'copy_node' : 'move_node' ]
+ (data.data.obj, lastmv.par, lastmv.pos);
+ }
+ })
+ .bind('keyup keydown', function (e) {
+ data = $.vakata.dnd._get();
+ if(data.data && data.data.jstree) {
+ data.helper.find('.jstree-copy:eq(0)')[ e[data.data.origin.get_settings().dnd.copy_modifier + "Key"] ? 'show' : 'hide' ]();
+ }
+ });
+
+ // add DND CSS
+ var css_string = '' +
+ '#jstree-marker { position: absolute; top:0; left:0; margin:0; padding:0; border-right:0; border-top:5px solid transparent; border-bottom:5px solid transparent; border-left:5px solid; width:0; height:0; font-size:0; line-height:0; _border-top-color:pink; _border-botton-color:pink; _filter:chroma(color=pink); } ' +
+ '#jstree-dnd { line-height:16px; margin:0; padding:4px; } ' +
+ '#jstree-dnd .jstree-icon, #jstree-dnd .jstree-copy { display:inline-block; text-decoration:none; margin:0 2px 0 0; padding:0; width:16px; height:16px; } ' +
+ '#jstree-dnd .jstree-ok { background:green; } ' +
+ '#jstree-dnd .jstree-er { background:red; } ' +
+ '#jstree-dnd .jstree-copy { margin:0 2px 0 2px; }';
+ $.vakata.css.add_sheet({ str : css_string, title : "jstree" });
+ });
+ // include the dnd plugin by default
+ $.jstree.defaults.plugins.push("dnd");
+})(jQuery); \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.hotkeys.js b/idrop-web/web-app/js/jstree.hotkeys.js
new file mode 100644
index 0000000..4acfce1
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.hotkeys.js
@@ -0,0 +1,138 @@
+/* File: jstree.hotkeys.js
+Enables keyboard shortcuts. Depends on jQuery.hotkeys (included).
+*/
+/* Group: jstree hotkeys plugin */
+(function ($) {
+ if(typeof $.hotkeys === "undefined") { throw "jsTree hotkeys: jQuery hotkeys plugin not included."; }
+
+ var bound = [];
+ function exec(i, event) {
+ var f = $.jstree._focused(), tmp;
+ if(f && f.data && f.data.hotkeys && f.data.hotkeys.enabled) {
+ tmp = f.get_settings(true).hotkeys[i];
+ if(tmp) { return tmp.call(f, event); }
+ }
+ }
+ $.jstree.plugin("hotkeys", {
+ __construct : function () {
+ if(!this.data.ui) { throw "jsTree hotkeys: jsTree UI plugin not included."; }
+ $.each(this.get_settings(true).hotkeys, function (i, v) {
+ if(v !== false && $.inArray(i, bound) == -1) {
+ $(document).bind("keydown", i, function (event) { return exec(i, event); });
+ bound.push(i);
+ }
+ });
+ this.get_container()
+ .bind("lock.jstree", $.proxy(function () {
+ if(this.data.hotkeys.enabled) { this.data.hotkeys.enabled = false; this.data.hotkeys.revert = true; }
+ }, this))
+ .bind("unlock.jstree", $.proxy(function () {
+ if(this.data.hotkeys.revert) { this.data.hotkeys.enabled = true; }
+ }, this));
+ this.enable_hotkeys();
+ },
+ defaults : {
+ "up" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected || -1;
+ this.hover_node(this.get_prev(o));
+ return false;
+ },
+ "ctrl+up" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected || -1;
+ this.hover_node(this.get_prev(o));
+ return false;
+ },
+ "shift+up" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected || -1;
+ this.hover_node(this.get_prev(o));
+ return false;
+ },
+ "down" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected || -1;
+ this.hover_node(this.get_next(o));
+ return false;
+ },
+ "ctrl+down" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected || -1;
+ this.hover_node(this.get_next(o));
+ return false;
+ },
+ "shift+down" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected || -1;
+ this.hover_node(this.get_next(o));
+ return false;
+ },
+ "left" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected;
+ if(o) {
+ if(o.hasClass("jstree-open")) { this.close_node(o); }
+ else { this.hover_node(this.get_prev(o)); }
+ }
+ return false;
+ },
+ "ctrl+left" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected;
+ if(o) {
+ if(o.hasClass("jstree-open")) { this.close_node(o); }
+ else { this.hover_node(this.get_prev(o)); }
+ }
+ return false;
+ },
+ "shift+left" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected;
+ if(o) {
+ if(o.hasClass("jstree-open")) { this.close_node(o); }
+ else { this.hover_node(this.get_prev(o)); }
+ }
+ return false;
+ },
+ "right" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected;
+ if(o && o.length) {
+ if(o.hasClass("jstree-closed")) { this.open_node(o); }
+ else { this.hover_node(this.get_next(o)); }
+ }
+ return false;
+ },
+ "ctrl+right" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected;
+ if(o && o.length) {
+ if(o.hasClass("jstree-closed")) { this.open_node(o); }
+ else { this.hover_node(this.get_next(o)); }
+ }
+ return false;
+ },
+ "shift+right" : function () {
+ var o = this.data.ui.hovered || this.data.ui.last_selected;
+ if(o && o.length) {
+ if(o.hasClass("jstree-closed")) { this.open_node(o); }
+ else { this.hover_node(this.get_next(o)); }
+ }
+ return false;
+ },
+ "space" : function () {
+ if(this.data.ui.hovered) { this.data.ui.hovered.children("a:eq(0)").click(); }
+ return false;
+ },
+ "ctrl+space" : function (event) {
+ event.type = "click";
+ if(this.data.ui.hovered) { this.data.ui.hovered.children("a:eq(0)").trigger(event); }
+ return false;
+ },
+ "shift+space" : function (event) {
+ event.type = "click";
+ if(this.data.ui.hovered) { this.data.ui.hovered.children("a:eq(0)").trigger(event); }
+ return false;
+ }
+ },
+ _fn : {
+ enable_hotkeys : function () {
+ this.data.hotkeys.enabled = true;
+ },
+ disable_hotkeys : function () {
+ this.data.hotkeys.enabled = false;
+ }
+ }
+ });
+ $.jstree.defaults.plugins.push("hotkeys");
+})(jQuery); \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.html.js b/idrop-web/web-app/js/jstree.html.js
new file mode 100644
index 0000000..27a22de
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.html.js
@@ -0,0 +1,69 @@
+/* File: jstree.html.js
+This plugin makes it possible for jstree to use HTML data sources (other than the container's initial HTML).
+*/
+/* Group: jstree html plugin */
+(function ($) {
+ $.jstree.plugin("html", {
+ defaults : {
+ data : false,
+ ajax : false
+ },
+ _fn : {
+ _append_html_data : function (dom, data) {
+ data = $(data);
+ if(!data || !data.length || !data.is('ul, li')) { return false; }
+ dom = this.get_node(dom);
+ if(dom === -1) { dom = this.get_container(); }
+ if(!dom.length) { return false; }
+ if(!dom.children('ul').length) { dom.append('<ul />'); }
+ dom.children('ul').empty().append(data.is('ul') ? data.children('li') : data);
+ return true;
+ },
+ _load_node : function (obj, callback) {
+ var d = false,
+ s = this.get_settings().html;
+ obj = this.get_node(obj);
+ if(!obj) { return false; }
+
+ switch(!0) {
+ // no settings - user original html
+ case (!s.data && !s.ajax):
+ if(obj === -1) {
+ this._append_html_data(-1, this.data.core.original_container_html.clone(true));
+ }
+ return callback.call(this, true);
+ // data is function
+ case ($.isFunction(s.data)):
+ return s.data.call(this, obj, $.proxy(function (d) {
+ return callback.call(this, this._append_html_data(obj, d));
+ }, this));
+ // data is set, ajax is not set, or both are set, but we are dealing with root node
+ case ((!!s.data && !s.ajax) || (!!s.data && !!s.ajax && obj === -1)):
+ return callback.call(this, this._append_html_data(obj, s.data));
+ // data is not set, ajax is set, or both are set, but we are dealing with a normal node
+ case ((!s.data && !!s.ajax) || (!!s.data && !!s.ajax && obj !== -1)):
+ s.ajax.success = $.proxy(function (d, t, x) {
+ var s = this.get_settings().html.ajax;
+ if($.isFunction(s.success)) {
+ d = s.success.call(this, d, t, x) || d;
+ }
+ callback.call(this, this._append_html_data(obj, d));
+ }, this);
+ s.ajax.error = $.proxy(function (x, t, e) {
+ var s = this.get_settings().html.ajax;
+ if($.isFunction(s.error)) {
+ s.error.call(this, x, t, e);
+ }
+ callback.call(this, false);
+ }, this);
+ if(!s.ajax.dataType) { s.ajax.dataType = "html"; }
+ if($.isFunction(s.ajax.url)) { s.ajax.url = s.ajax.url.call(this, obj); }
+ if($.isFunction(s.ajax.data)) { s.ajax.data = s.ajax.data.call(this, obj); }
+ return $.ajax(s.ajax);
+ }
+ }
+ }
+ });
+ // include the html plugin by default
+ $.jstree.defaults.plugins.push("html");
+})(jQuery); \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.json.js b/idrop-web/web-app/js/jstree.json.js
new file mode 100644
index 0000000..03683dc
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.json.js
@@ -0,0 +1,99 @@
+/* File: jstree.json.js
+This plugin makes it possible for jstree to use JSON data sources.
+*/
+/* Group: jstree json plugin */
+(function ($) {
+ $.jstree.plugin("json", {
+ __construct : function () {
+ this.get_container()
+ .bind("__after_close.jstree", $.proxy(function (e, data) {
+ var t = $(data.rslt.obj);
+ if(this.get_settings(true).json.progressive_unload) {
+ t.data('jstree').children = this.get_json(t)[0].children;
+ t.children("ul").remove();
+ }
+ }, this));
+ },
+ defaults : {
+ data : false,
+ ajax : false,
+ progressive_render : false, // get_json, data on each node
+ progressive_unload : false
+ },
+ _fn : {
+ parse_json : function (node) {
+ var s = this.get_settings(true).json;
+ if($.isArray(node.children)) {
+ if(s.progressive_render) {
+ if(!node.data) { node.data = {}; }
+ if(!node.data.jstree) { node.data.jstree = {}; }
+ node.data.jstree.children = node.children;
+ node.children = true;
+ }
+ }
+ return this.__call_old(true, node);
+ },
+ _append_json_data : function (dom, data) {
+ dom = this.get_node(dom);
+ if(dom === -1) { dom = this.get_container(); }
+ data = this.parse_json(data);
+ if(!data || !dom.length) { return false; }
+ if(!dom.children('ul').length) { dom.append('<ul />'); }
+ dom.children('ul').empty().append(data.children('li'));
+ return true;
+ },
+ _load_node : function (obj, callback) {
+ var d = false,
+ s = this.get_settings().json;
+ obj = this.get_node(obj);
+ if(!obj) { return false; }
+
+ switch(!0) {
+ // root node with data
+ case (obj === -1 && this.get_container().data('jstree') && $.isArray(this.get_container().data('jstree').children)):
+ d = this.get_container().data('jstree').children;
+ this.get_container().data('jstree').children = null;
+ return callback.call(this, this._append_json_data(obj, d));
+ // normal node with data
+ case (obj !== -1 && obj.length && obj.data('jstree') && $.isArray(obj.data('jstree').children)):
+ d = obj.data('jstree').children;
+ obj.data('jstree').children = null;
+ return callback.call(this, this._append_json_data(obj, d));
+ // no settings
+ case (!s.data && !s.ajax):
+ throw "Neither data nor ajax settings supplied.";
+ // data is function
+ case ($.isFunction(s.data)):
+ return s.data.call(this, obj, $.proxy(function (d) {
+ return callback.call(this, this._append_json_data(obj, d));
+ }, this));
+ // data is set, ajax is not set, or both are set, but we are dealing with root node
+ case ((!!s.data && !s.ajax) || (!!s.data && !!s.ajax && obj === -1)):
+ return callback.call(this, this._append_json_data(obj, s.data));
+ // data is not set, ajax is set, or both are set, but we are dealing with a normal node
+ case ((!s.data && !!s.ajax) || (!!s.data && !!s.ajax && obj !== -1)):
+ s.ajax.success = $.proxy(function (d, t, x) {
+ var s = this.get_settings().json.ajax;
+ if($.isFunction(s.success)) {
+ d = s.success.call(this, d, t, x) || d;
+ }
+ callback.call(this, this._append_json_data(obj, d));
+ }, this);
+ s.ajax.error = $.proxy(function (x, t, e) {
+ var s = this.get_settings().json.ajax;
+ if($.isFunction(s.error)) {
+ s.error.call(this, x, t, e);
+ }
+ callback.call(this, false);
+ }, this);
+ if(!s.ajax.dataType) { s.ajax.dataType = "json"; }
+ if($.isFunction(s.ajax.url)) { s.ajax.url = s.ajax.url.call(this, obj); }
+ if($.isFunction(s.ajax.data)) { s.ajax.data = s.ajax.data.call(this, obj); }
+ return $.ajax(s.ajax);
+ }
+ }
+ }
+ });
+ // include the json plugin by default
+ // $.jstree.defaults.plugins.push("json");
+})(jQuery); \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.sort.js b/idrop-web/web-app/js/jstree.sort.js
new file mode 100644
index 0000000..ba55df8
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.sort.js
@@ -0,0 +1,38 @@
+/* File: jstree.sort.js
+Sorts items alphabetically (or using any other function)
+*/
+/* Group: jstree sort plugin */
+(function ($) {
+ $.jstree.plugin("sort", {
+ __construct : function () {
+ this.get_container()
+ .bind("load_node.jstree", $.proxy(function (e, data) {
+ var obj = this.get_node(data.rslt.obj);
+ obj = obj === -1 ? this.get_container_ul() : obj.children("ul");
+ this._sort(obj, true);
+ }, this))
+ .bind("rename_node.jstree create_node.jstree", $.proxy(function (e, data) {
+ this._sort(data.rslt.obj.parent(), false);
+ }, this))
+ .bind("move_node.jstree copy_node.jstree", $.proxy(function (e, data) {
+ var m = data.rslt.parent == -1 ? this.get_container_ul() : data.rslt.parent.children('ul');
+ this._sort(m, false);
+ }, this));
+ },
+ defaults : function (a, b) { return this.get_text(a, true) > this.get_text(b, true) ? 1 : -1; },
+ _fn : {
+ _sort : function (obj, deep) {
+ var s = this.get_settings(true).sort,
+ t = this;
+ obj.append($.makeArray(obj.children("li")).sort($.proxy(s, t)));
+ obj.children('li').each(function () { t.correct_node(this, false); });
+ if(deep) {
+ obj.find("> li > ul").each(function() { t._sort($(this)); });
+ t.correct_node(obj.children('li'), true);
+ }
+ }
+ }
+ });
+ // include the sort plugin by default
+ $.jstree.defaults.plugins.push("sort");
+})(jQuery); \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.state.js b/idrop-web/web-app/js/jstree.state.js
new file mode 100644
index 0000000..16fd207
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.state.js
@@ -0,0 +1,39 @@
+/* File: jstree.state.js
+This plugin enables state saving between reloads.
+*/
+/* Group: jstree state plugin */
+(function ($) {
+ $.jstree.plugin("state", {
+ __construct : function () {
+ if(typeof $.vakata.storage === "undefined") { throw "jsTree state plugin: vakata storage helper not included."; }
+
+ this.get_container()
+ .bind("__loaded.jstree", $.proxy(function (e, data) {
+ this.restore_state();
+ }, this))
+ .bind("__ready.jstree", $.proxy(function (e, data) {
+ this.get_container()
+ .bind(this.get_settings(true).state.events, $.proxy(function () {
+ this.save_state();
+ }, this));
+ }, this));
+ },
+ defaults : {
+ key : 'jstree', // pass unique name to work with many trees
+ events : 'select_node.jstree open_node.jstree close_node.jstree deselect_node.jstree deselect_all.jstree'
+ },
+ _fn : {
+ save_state : function () {
+ var s = this.get_settings(true).state;
+ $.vakata.storage.set(s.key, this.get_state());
+ },
+ restore_state : function () {
+ var s = this.get_settings(true).state,
+ k = $.vakata.storage.get(s.key);
+ if(!!k) { this.set_state(k); }
+ }
+ }
+ });
+ // include the state plugin by default
+ // $.jstree.defaults.plugins.push("state");
+})(jQuery);
diff --git a/idrop-web/web-app/js/jstree.themes.js b/idrop-web/web-app/js/jstree.themes.js
new file mode 100644
index 0000000..8c08089
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.themes.js
@@ -0,0 +1,215 @@
+/* File: jstree.themes.js
+Controls the looks of jstree, without this plugin you will get a functional tree, but it will look just like an ordinary UL list
+*/
+(function ($) {
+ var themes_loaded = [];
+ /*
+ Group: $.jstree.
+
+ Variable: $.jstree.THEMES_DIR
+ The location of all themes, this is used when setting a theme without supplying an URL (only by name).
+ Default is _false_. If left as _false_ the path will be autodetected when the DOM is ready.
+ The location of _jquery.jstree.js_ is used for the autodetection.
+ Normally you won't need to modify this (provided you leave the _themes_ folder in the same folder as _jquery.jstree.js_ and do not rename the file).
+ If you decide to move the folder or rename the file, but still want to load themes by name, simply set this to the new location of the _themes_ folder.
+ > <script type="text/javascript" src="jquery.jstree.js"></script>
+ > <script type="text/javascript">$.jstree.THEMES_DIR = "some/path/with-a-trailing-slash/";</script>
+ */
+ $.jstree.THEMES_DIR = false;
+
+ $.jstree.plugin("themes", {
+ __construct : function () {
+ this.get_container()
+ .bind("__construct.jstree", $.proxy(function () {
+ var s = this.get_settings(true).themes;
+ this.data.themes.dots = s.dots;
+ this.data.themes.icons = s.icons;
+
+ if(s.url === false && s.theme === false) {
+ s.theme = this.data.core.rtl ? 'default-rtl' : 'default';
+ }
+ this.set_theme(s.theme, s.url);
+
+ this[ this.data.themes.dots ? "show_dots" : "hide_dots" ]();
+ this[ this.data.themes.icons ? "show_icons" : "hide_icons" ]();
+ }, this));
+ },
+ /* Class: jstree */
+ /*
+ Group: THEMES options
+
+ Variable: config.themes.theme
+ *string* the name of the theme you want to use. Default is _default_.
+
+ Variable: config.themes.url
+ *mixed* the URL of the stylesheet of the theme you want to use. Default is _false_. If left as _false_ the location will be autodetected using <$.jstree.THEMES_DIR>.
+
+ Variable: config.themes.dots
+ *boolean* whether to show dots or not. Default is _true_. The chosen theme should support this option.
+
+ Variable: config.themes.icons
+ *boolean* whether to show icons or not. Default is _true_.
+ */
+ defaults : {
+ theme : false,
+ url : false,
+ dots : true,
+ icons : true
+ },
+ _fn : {
+ /*
+ Group: THEMES functions
+
+ Function: set_theme
+ Sets the tree theme. This function is automatically called at construction with the settings specified in <config.themes.theme> and <config.themes.theme.url>.
+
+ Parameters:
+ theme_name - the name of the theme to apply
+ theme_url - the URL of the stylesheet - leave this blank for autodetect
+
+ Example:
+ >// Set the theme and autodetect the location
+ >$("#div1").jstree("set_theme","classic");
+ >// A custom theme. Please note that if you place your own theme in the _themes_ folder ot will be autodetected too.
+ >$("#div2").jstree("set_theme","custom-theme","/some/path/theme.css");
+ */
+ set_theme : function (theme_name, theme_url) {
+ if(!theme_name) { return false; }
+ if(!theme_url) { theme_url = $.jstree.THEMES_DIR + theme_name + '/style.css'; }
+ if($.inArray(theme_url, themes_loaded) === -1) {
+ $.vakata.css.add_sheet({ "url" : theme_url });
+ themes_loaded.push(theme_url);
+ }
+ if(this.data.themes.theme != theme_name) {
+ this.get_container().removeClass('jstree-' + this.data.themes.theme);
+ this.data.themes.theme = theme_name;
+ }
+ this.get_container().addClass('jstree-' + theme_name);
+ this.__callback(theme_name);
+ },
+ get_theme : function () { return this.data.themes.theme; },
+ show_dots : function () { this.data.themes.dots = true; this.get_container().children("ul").removeClass("jstree-no-dots"); },
+ hide_dots : function () { this.data.themes.dots = false; this.get_container().children("ul").addClass("jstree-no-dots"); },
+ toggle_dots : function () { if(this.data.themes.dots) { this.hide_dots(); } else { this.show_dots(); } },
+ show_icons : function () { this.data.themes.icons = true; this.get_container().children("ul").removeClass("jstree-no-icons"); },
+ hide_icons : function () { this.data.themes.icons = false; this.get_container().children("ul").addClass("jstree-no-icons"); },
+ toggle_icons : function () { if(this.data.themes.icons) { this.hide_icons(); } else { this.show_icons(); } },
+
+ set_icon : function (obj, icon) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ obj = obj.find("> a > .jstree-themeicon");
+ if(icon === false) {
+ this.hide_icon(obj);
+ }
+ else if(icon.indexOf("/") === -1) {
+ obj.addClass(icon).attr("rel",icon);
+ }
+ else {
+ obj.css("background", "url('" + icon + "') center center no-repeat").attr("rel",icon);
+ }
+ return true;
+ },
+ get_icon : function (obj) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length) { return null; }
+ obj = obj.find("> a > .jstree-themeicon");
+ if(obj.hasClass('jstree-themeicon-hidden')) { return false; }
+ obj = obj.attr("rel");
+ return (obj && obj.length) ? obj : null;
+ },
+ hide_icon : function (obj) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ obj.find('> a > .jstree-themeicon').addClass('jstree-themeicon-hidden');
+ return true;
+ },
+ show_icon : function (obj) {
+ obj = this.get_node(obj);
+ if(!obj || obj === -1 || !obj.length) { return false; }
+ obj.find('> a > .jstree-themeicon').removeClass('jstree-themeicon-hidden');
+ return true;
+ },
+
+ clean_node : function(obj) {
+ obj = this.__call_old();
+ var t = this;
+ return obj.each(function () {
+ var o = $(this),
+ d = o.data("jstree");
+ if(!o.find("> a > ins.jstree-themeicon").length) {
+ o.children("a").prepend("<ins class='jstree-icon jstree-themeicon'>&#160;</ins>");
+ }
+ if(d && typeof d.icon !== 'undefined') {
+ t.set_icon(o, d.icon);
+ delete d.icon;
+ }
+ });
+ },
+ get_state : function () {
+ var state = this.__call_old();
+ state.themes = { 'theme' : this.get_theme(), 'icons' : this.data.themes.icons, 'dots' : this.data.themes.dots };
+ return state;
+ },
+ set_state : function (state, callback) {
+ if(this.__call_old()) {
+ if(state.themes) {
+ if(state.themes.theme) {
+ this.set_theme(state.themes.theme);
+ }
+ if(typeof state.themes.dots !== 'undefined') {
+ this[ state.themes.dots ? "show_dots" : "hide_dots" ]();
+ }
+ if(typeof state.themes.icons !== 'undefined') {
+ this[ state.themes.icons ? "show_icons" : "hide_icons" ]();
+ }
+ delete state.themes;
+ this.set_state(state, callback);
+ return false;
+ }
+ return true;
+ }
+ return false;
+ },
+ get_json : function (obj, is_callback) {
+ var r = this.__call_old(), i;
+ if(is_callback) {
+ i = this.get_icon(obj);
+ if(typeof i !== 'undefined' && i !== null) {
+ r.data.jstree.icon = i;
+ }
+ }
+ return r;
+ }
+ }
+ });
+ $(function () {
+ // autodetect themes path
+ if($.jstree.THEMES_DIR === false) {
+ $("script").each(function () {
+ if(this.src.toString().match(/jquery\.jstree[^\/]*?\.js(\?.*)?$/)) {
+ $.jstree.THEMES_DIR = this.src.toString().replace(/jquery\.jstree[^\/]*?\.js(\?.*)?$/, "") + 'themes/';
+ return false;
+ }
+ });
+ }
+ if($.jstree.THEMES_DIR === false) { $.jstree.THEMES_DIR = "themes/"; }
+ // add themes specific CSS
+ var css_string = '' +
+ '.jstree a { text-decoration:none; } ' +
+ '.jstree a > .jstree-themeicon { height:16px; width:16px; margin-right:3px; } ' +
+ '.jstree-rtl a > .jstree-themeicon { margin-left:3px; margin-right:0; } ' +
+ '.jstree .jstree-no-icons .jstree-themeicon, .jstree .jstree-themeicon-hidden { display:none; } ';
+ // Correct IE 6 (does not support the > CSS selector)
+ if($.jstree.IS_IE6) {
+ css_string += '' +
+ '.jstree li a .jstree-themeicon { height:16px; width:16px; margin-right:3px; } ' +
+ '.jstree-rtl li a .jstree-themeicon { margin-right:0px; margin-left:3px; } ';
+ }
+ // the default stylesheet
+ $.vakata.css.add_sheet({ str : css_string, title : "jstree" });
+ });
+ // include the themes plugin by default
+ $.jstree.defaults.plugins.push("themes");
+})(jQuery);
+//*/ \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.ui.js b/idrop-web/web-app/js/jstree.ui.js
new file mode 100644
index 0000000..663c8a6
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.ui.js
@@ -0,0 +1,201 @@
+/* File: jstree.ui.js
+This plugin enables selecting, deselecting and hovering tree items.
+*/
+/* Group: jstree UI plugin */
+(function ($) {
+ $.jstree.plugin("ui", {
+ __construct : function () {
+ this.data.ui.selected = $();
+ this.data.ui.hovered = null;
+ this.data.ui.last_selected = false;
+
+ this.get_container() // TODO: configurable event (click/dblclick/etc)
+ .delegate("a", "click.jstree", $.proxy(function (e) {
+ e.preventDefault();
+ e.currentTarget.blur();
+ var s = this.get_settings(true).ui,
+ obj = this.get_node(e.currentTarget),
+ is_selected = this.is_selected(obj),
+ is_multiple = s.select_multiple_modifier == "on" || (s.select_multiple_modifier !== false && e && e[s.select_multiple_modifier + "Key"]),
+ is_range = s.select_multiple_modifier == "on" || (s.select_range_modifier !== false && e && e[s.select_range_modifier + "Key"] && this.data.ui.last_selected && this.data.ui.last_selected[0] !== obj[0] && this.data.ui.last_selected.parent()[0] === obj.parent()[0]);
+
+ switch(!0) {
+ case (is_range && this.data.ui.last_selected !== false):
+ this.select_range(obj);
+ break;
+ case (is_range && this.data.ui.last_selected === false):
+ this.select_one(obj);
+ break;
+ case (is_selected && is_multiple):
+ this.deselect_node(obj);
+ break;
+ default:
+ this.select_one(obj, is_multiple);
+ break;
+ }
+ }, this))
+ .delegate("a", "mouseenter.jstree", $.proxy(function (e) {
+ this.hover_node(e.target);
+ }, this))
+ .delegate("a", "mouseleave.jstree", $.proxy(function (e) {
+ this.dehover_node(e.target);
+ }, this))
+ .bind("delete_node.jstree", $.proxy(function (event, data) {
+ var o = this.get_node(data.rslt.obj),
+ n = (o && o.length) ? o.find("a.jstree-clicked") : $(),
+ t = this;
+ n.each(function () { t.deselect_node(this); });
+ }, this))
+ .bind("move_node.jstree", $.proxy(function (event, data) {
+ if(data.rslt.cy) {
+ data.rslt.oc.find("a.jstree-clicked").removeClass("jstree-clicked");
+ }
+ }, this));
+ },
+ defaults : {
+ select_multiple_modifier : "ctrl", // on, or ctrl, shift, alt, or false
+ select_range_modifier : "shift", // on, or ctrl, shift, alt, or false
+ disable_nested_selection : true
+ },
+ _fn : {
+ get_node : function (obj, allow_multiple) {
+ if(typeof obj === "undefined" || obj === null) { return allow_multiple ? this.data.ui.selected : this.data.ui.last_selected; }
+ return this.__call_old();
+ },
+
+ hover_node : function (obj) {
+ obj = this.get_node(obj);
+ if(!obj || !obj.length || this.is_loading(obj)) { return false; }
+ if(!obj.hasClass("jstree-hovered")) { this.dehover_node(); }
+ this.data.ui.hovered = obj.children("a").addClass("jstree-hovered").parent();
+ this.scroll_to_node(obj);
+ this.__callback({ "obj" : obj });
+ },
+ dehover_node : function () {
+ var obj = this.data.ui.hovered, p;
+ if(!obj || !obj.length) { return false; }
+ p = obj.children("a").removeClass("jstree-hovered").parent();
+ if(this.data.ui.hovered[0] === p[0]) { this.data.ui.hovered = null; }
+ this.__callback({ "obj" : obj });
+ },
+ select_node : function (obj) {
+ var t = this;
+ obj = this.get_node(obj);
+ if(obj == -1 || !obj || !obj.length || this.is_loading(obj)) { return false; }
+ obj.children("a").addClass("jstree-clicked");
+ this.data.ui.last_selected = obj;
+ this.data.ui.selected = this.data.ui.selected.add(obj);
+ // this.scroll_to_node(obj.eq(0));
+ obj.parents(".jstree-closed").each(function () { t.open_node(this, false, 0); });
+ this.__callback({ "obj" : obj });
+ },
+ deselect_node : function (obj) {
+ obj = this.get_node(obj);
+ if(!obj || !obj.length) { return false; }
+ if(this.is_selected(obj)) {
+ obj.children("a").removeClass("jstree-clicked");
+ this.data.ui.selected = this.data.ui.selected.not(obj);
+ if(this.data.ui.last_selected.get(0) === obj.get(0)) { this.data.ui.last_selected = this.data.ui.selected.eq(0); }
+ this.__callback({ "obj" : obj });
+ }
+ },
+ deselect_all : function (context) {
+ var ret = context ? $(context).find("a.jstree-clicked").parent() : this.get_container().find("a.jstree-clicked").parent();
+ ret.children("a.jstree-clicked").removeClass("jstree-clicked");
+ this.data.ui.selected = $();
+ this.data.ui.last_selected = false;
+ this.__callback({ "obj" : ret });
+ },
+ is_selected : function (obj) { return this.data.ui.selected.index(this.get_node(obj)) >= 0; },
+ get_selected : function (context) { return context ? $(context).find("a.jstree-clicked").parent() : this.data.ui.selected; },
+
+ select_range : function (obj, start_node, keep_old_selection) {
+ var _this = this, i, s;
+ obj = this.get_node(obj);
+ if(!start_node) { s = true; start_node = this.data.ui.last_selected; }
+ start_node = this.get_node(start_node);
+ if(obj == -1 || !obj || !obj.length || this.is_loading(obj)) { return false; }
+ if(start_node == -1 || !start_node || !start_node.length || this.is_loading(start_node)) { return false; }
+
+ if(!keep_old_selection) { this.deselect_all(); }
+ i = (obj.index() < start_node.index());
+ start_node.addClass("jstree-last-selected");
+ obj = obj[ i ? "nextUntil" : "prevUntil" ](".jstree-last-selected").andSelf().add(".jstree-last-selected");
+ start_node.removeClass("jstree-last-selected");
+ if(!i) { obj = obj.vakata_reverse(); }
+ if(!obj.length) { return false; }
+ obj.each(function () { _this.select_node(this); });
+ if(s) { this.data.ui.last_selected = start_node; }
+ this.__callback({ "obj" : obj });
+ return true;
+ },
+ select_one : function (obj, keep_old_selection) {
+ obj = this.get_node(obj);
+ if(obj == -1 || !obj || !obj.length || this.is_loading(obj)) { return false; }
+ if(!keep_old_selection) { this.deselect_all(); }
+ else {
+ if(
+ this.get_settings(true).ui.disable_nested_selection &&
+ (
+ (obj.parentsUntil(".jstree","li").children("a.jstree-clicked:eq(0)").length) ||
+ (obj.children("ul").find("a.jstree-clicked:eq(0)").length)
+ )
+ ) {
+ return false;
+ }
+ }
+ this.select_node(obj);
+ // obj.each(function () { t.select_node(this); });
+ this.__callback({ "obj" : obj });
+ return true;
+ },
+
+ clean_node : function(obj) {
+ obj = this.__call_old();
+ var _this = this;
+ return obj.each(function () {
+ var t = $(this),
+ d = t.data("jstree");
+ t.find('.jstree-clicked').removeClass('jstree-clicked');
+ if(d && d.selected) {
+ _this.select_node(t);
+ delete d.selected;
+ }
+ });
+ },
+ get_state : function () {
+ var state = this.__call_old();
+ state.selected = [];
+ this.data.ui.selected.each(function () { state.selected.push(this.id); });
+ return state;
+ },
+ set_state : function (state, callback) {
+ if(this.__call_old()) {
+ if(state.selected) {
+ var _this = this;
+ this.deselect_all();
+ $.each(state.selected, function (i, v) {
+ _this.select_node(document.getElementById(v));
+ });
+ delete state.selected;
+ this.set_state(state, callback);
+ return false;
+ }
+ return true;
+ }
+ return false;
+ },
+ get_json : function (obj, is_callback) {
+ var r = this.__call_old();
+ if(is_callback) {
+ if(this.is_selected(obj)) {
+ r.data.jstree.selected = true;
+ }
+ }
+ return r;
+ }
+ }
+ });
+ // include the selection plugin by default
+ $.jstree.defaults.plugins.push("ui");
+})(jQuery); \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.unique.js b/idrop-web/web-app/js/jstree.unique.js
new file mode 100644
index 0000000..8e76c22
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.unique.js
@@ -0,0 +1,33 @@
+/* File: jstree.unique.js
+Does not allow the same name amongst siblings (still a bit experimental).
+*/
+/* Group: jstree drag'n'drop plugin */
+(function ($) {
+ $.jstree.plugin("unique", {
+ // TODO: think about an option to work with HTML or not?
+ _fn : {
+ check : function (chk, obj, par, pos) {
+ if(!this.__call_old()) { return false; }
+
+ par = par === -1 ? this.get_container() : par;
+ var n = chk === "rename_node" ? $('<div />').html(pos).text() : this.get_text(obj, true),
+ c = [],
+ t = this;
+ par.children('ul').children('li').each(function () { c.push(t.get_text(this, true)); });
+ switch(chk) {
+ case "delete_node":
+ return true;
+ case "rename_node":
+ case "copy_node":
+ return ($.inArray(n, c) === -1);
+ case "move_node":
+ return (par.children('ul').children('li').index(obj) !== -1 || $.inArray(n, c) === -1);
+ }
+ return true;
+ }
+ }
+ });
+ // include the unique plugin by default
+ $.jstree.defaults.plugins.push("unique");
+})(jQuery);
+//*/ \ No newline at end of file
diff --git a/idrop-web/web-app/js/jstree.xml.js b/idrop-web/web-app/js/jstree.xml.js
new file mode 100644
index 0000000..d7a3e31
--- /dev/null
+++ b/idrop-web/web-app/js/jstree.xml.js
@@ -0,0 +1,185 @@
+/* File: jstree.xml.js
+This plugin makes it possible for jstree to use XML data sources.
+*/
+/* Group: jstree xml plugin */
+(function ($) {
+ var xsl = {
+ 'nest' : '' +
+ '<' + '?xml version="1.0" encoding="utf-8" ?>' +
+ '<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" >' +
+ '<xsl:output method="html" encoding="utf-8" omit-xml-declaration="yes" standalone="no" indent="no" media-type="text/html" />' +
+ '<xsl:template match="/">' +
+ ' <xsl:call-template name="nodes">' +
+ ' <xsl:with-param name="node" select="/root" />' +
+ ' </xsl:call-template>' +
+ '</xsl:template>' +
+ '<xsl:template name="nodes">' +
+ ' <xsl:param name="node" />' +
+ ' <ul>' +
+ ' <xsl:for-each select="$node/item">' +
+ ' <xsl:variable name="children" select="count(./item) &gt; 0" />' +
+ ' <li>' +
+ ' <xsl:for-each select="@*"><xsl:attribute name="{name()}"><xsl:value-of select="." /></xsl:attribute></xsl:for-each>' +
+ ' <a>' +
+ ' <xsl:for-each select="./content/@*"><xsl:attribute name="{name()}"><xsl:value-of select="." /></xsl:attribute></xsl:for-each>' +
+ ' <xsl:copy-of select="./content/child::node()" />' +
+ ' </a>' +
+ ' <xsl:if test="$children"><xsl:call-template name="nodes"><xsl:with-param name="node" select="current()" /></xsl:call-template></xsl:if>' +
+ ' </li>' +
+ ' </xsl:for-each>' +
+ ' </ul>' +
+ '</xsl:template>' +
+ '</xsl:stylesheet>',
+ 'flat' : '' +
+ '<' + '?xml version="1.0" encoding="utf-8" ?>' +
+ '<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" >' +
+ '<xsl:output method="html" encoding="utf-8" omit-xml-declaration="yes" standalone="no" indent="no" media-type="text/xml" />' +
+ '<xsl:template match="/">' +
+ ' <ul>' +
+ ' <xsl:for-each select="//item[not(@parent_id) or @parent_id=0 or not(@parent_id = //item/@id)]">' + /* the last `or` may be removed */
+ ' <xsl:call-template name="nodes">' +
+ ' <xsl:with-param name="node" select="." />' +
+ ' </xsl:call-template>' +
+ ' </xsl:for-each>' +
+ ' </ul>' +
+ '</xsl:template>' +
+ '<xsl:template name="nodes">' +
+ ' <xsl:param name="node" />' +
+ ' <xsl:variable name="children" select="count(//item[@parent_id=$node/attribute::id]) &gt; 0" />' +
+ ' <li>' +
+ ' <xsl:for-each select="@*">' +
+ ' <xsl:if test="name() != \'parent_id\'">' +
+ ' <xsl:attribute name="{name()}"><xsl:value-of select="." /></xsl:attribute>' +
+ ' </xsl:if>' +
+ ' </xsl:for-each>' +
+ ' <a>' +
+ ' <xsl:for-each select="./content/@*"><xsl:attribute name="{name()}"><xsl:value-of select="." /></xsl:attribute></xsl:for-each>' +
+ ' <xsl:copy-of select="./content/child::node()" />' +
+ ' </a>' +
+ ' <xsl:if test="$children">' +
+ ' <ul>' +
+ ' <xsl:for-each select="//item[@parent_id=$node/attribute::id]">' +
+ ' <xsl:call-template name="nodes">' +
+ ' <xsl:with-param name="node" select="." />' +
+ ' </xsl:call-template>' +
+ ' </xsl:for-each>' +
+ ' </ul>' +
+ ' </xsl:if>' +
+ ' </li>' +
+ '</xsl:template>' +
+ '</xsl:stylesheet>'
+ },
+ escape_xml = function(string) {
+ return string
+ .toString()
+ .replace(/&/g, '&amp;')
+ .replace(/</g, '&lt;')
+ .replace(/>/g, '&gt;')
+ .replace(/"/g, '&quot;')
+ .replace(/'/g, '&apos;');
+ };
+
+
+ $.jstree.plugin("xml", {
+ defaults : {
+ xsl : "flat",
+ data : false,
+ ajax : false
+ },
+ _fn : {
+ _append_xml_data : function (dom, data) {
+ data = $.vakata.xslt(data, xsl[this.get_settings().xml.xsl]);
+ if(data === false) { return false; }
+ data = $(data);
+ if(!data || !data.length || !data.is('ul, li')) { return false; }
+ dom = this.get_node(dom);
+ if(dom === -1) { dom = this.get_container(); }
+ if(!dom.length) { return false; }
+ if(!dom.children('ul').length) { dom.append('<ul />'); }
+ dom.children('ul').empty().append(data.is('ul') ? data.children('li') : data);
+ return true;
+ },
+ _load_node : function (obj, callback) {
+ var d = false,
+ s = this.get_settings().xml;
+ obj = this.get_node(obj);
+ if(!obj) { return false; }
+ switch(!0) {
+ // data is function
+ case ($.isFunction(s.data)):
+ return s.data.call(this, obj, $.proxy(function (d) {
+ return callback.call(this, this._append_xml_data(obj, d));
+ }, this));
+ // data is set, ajax is not set, or both are set, but we are dealing with root node
+ case ((!!s.data && !s.ajax) || (!!s.data && !!s.ajax && obj === -1)):
+ return callback.call(this, this._append_xml_data(obj, s.data));
+ // data is not set, ajax is set, or both are set, but we are dealing with a normal node
+ case ((!s.data && !!s.ajax) || (!!s.data && !!s.ajax && obj !== -1)):
+ s.ajax.success = $.proxy(function (d, t, x) {
+ var s = this.get_settings().xml.ajax;
+ if($.isFunction(s.success)) {
+ d = s.success.call(this, d, t, x) || d;
+ }
+ callback.call(this, this._append_xml_data(obj, d));
+ }, this);
+ s.ajax.error = $.proxy(function (x, t, e) {
+ var s = this.get_settings().xml.ajax;
+ if($.isFunction(s.error)) {
+ s.error.call(this, x, t, e);
+ }
+ callback.call(this, false);
+ }, this);
+ if(!s.ajax.dataType) { s.ajax.dataType = "xml"; }
+ if($.isFunction(s.ajax.url)) { s.ajax.url = s.ajax.url.call(this, obj); }
+ if($.isFunction(s.ajax.data)) { s.ajax.data = s.ajax.data.call(this, obj); }
+ return $.ajax(s.ajax);
+ }
+ },
+ get_xml : function (mode, obj, is_callback) {
+ var r = '';
+ if(!mode) { mode = 'flat'; }
+ if(typeof is_callback === 'undefined') {
+ obj = this.get_json(obj);
+ $.each(obj, $.proxy(function (i, v) {
+ r += this.get_xml(mode, v, true);
+ }, this));
+ return '' +
+ '<' + '?xml version="1.0" encoding="utf-8" ?>' +
+ '<root>' + r + '</root>';
+ }
+ r += '<item';
+ if(mode === 'flat' && is_callback !== true) {
+ r += ' parent_id="' + escape_xml(is_callback) + '"';
+ }
+ if(obj.data && !$.isEmptyObject(obj.data)) {
+ $.each(obj.data, function (i, v) {
+ if(!$.isEmptyObject(v)) {
+ r += ' data-' + i + '="' + escape_xml($.vakata.json.encode(v)) + '"';
+ }
+ });
+ }
+ $.each(obj.li_attr, function (i, v) {
+ r += ' ' + i + '="' + escape_xml(v) + '"';
+ });
+ r += '>';
+ r += '<content';
+ $.each(obj.a_attr, function (i, v) {
+ r += ' ' + i + '="' + escape_xml(v) + '"';
+ });
+ r += '><![CDATA[' + obj.title + ']]></content>';
+
+ if(mode === 'flat') { r += '</item>'; }
+ if(obj.children) {
+ $.each(obj.children, $.proxy(function (i, v) {
+ r += this.get_xml(mode, v, obj.li_attr && obj.li_attr.id ? obj.li_attr.id : true);
+ }, this));
+ }
+ if(mode === 'nest') { r += '</item>'; }
+ return r;
+ }
+ }
+ });
+ // include the html plugin by default
+ $.jstree.defaults.plugins.push("xml");
+})(jQuery);
+//*/ \ No newline at end of file
diff --git a/idrop-web/web-app/js/vakata.js b/idrop-web/web-app/js/vakata.js
new file mode 100644
index 0000000..1958fd1
--- /dev/null
+++ b/idrop-web/web-app/js/vakata.js
@@ -0,0 +1,1930 @@
+/*
+File: Helper functions
+This file includes some functions that enable CSS manipulations, contextmenus, XSLT transformations and drag'n'drop.
+All of those work independently of jstree.
+*/
+
+/*
+Variable: $.vakata
+*object* Holds all helper objects.
+*/
+(function ($) {
+ $.vakata = {};
+})(jQuery);
+
+/*
+Group: Miscellaneous
+Various small snippets.
+*/
+
+/*
+Function: $().vakata_reverse
+Makes it possible to apply the standard array reverse function to a jQuery collection.
+
+Input:
+> <div>1</div><div>2</div><div>3</div>
+> $("div").vakata_reverse().each(function () { document.write(this.innerHTML); });
+
+Output:
+>321
+*/
+(function ($) {
+ $.fn.vakata_reverse = [].reverse;
+})(jQuery);
+
+/*
+Function: $.vakata.array_remove
+Makes it possible to remove an item (or a group of items) form an array.
+http://ejohn.org/blog/javascript-array-remove/
+
+Input:
+> $.vakata.array_remove(['a', 'b', 'c'], 1);
+
+Output:
+>['a', 'c']
+*/
+(function ($) {
+ $.vakata.array_remove = function(array, from, to) {
+ var rest = array.slice((to || from) + 1 || array.length);
+ array.length = from < 0 ? array.length + from : from;
+ array.push.apply(array, rest);
+ return array;
+ };
+})(jQuery);
+
+/*
+Function: $.vakata.array_unique
+Returns only the unique items from an array.
+
+Input:
+> $.vakata.array_unique(['c','a','a','b','c','b']);
+
+Output:
+>['a', 'b', 'c']
+*/
+(function ($) {
+ $.vakata.array_unique = function(array) {
+ var a = [], i, j, l;
+ for(i = 0, l = array.length; i < l; i++) {
+ for(j = 0; j <= i; j++) {
+ if(array[i] === array[j]) {
+ break;
+ }
+ }
+ if(j === i) { a.push(array[i]); }
+ }
+ return a;
+ };
+})(jQuery);
+
+/*
+Function: $.vakata.attributes
+Collects all attributes from a DOM node.
+*/
+(function ($) {
+ $.vakata.attributes = function(node, with_values) {
+ node = $(node)[0];
+ var attr = with_values ? {} : [];
+ $.each(node.attributes, function (i, v) {
+ if($.inArray(v.nodeName.toLowerCase(),['style','contenteditable','hasfocus','tabindex']) !== -1) { return; }
+ if(v.nodeValue !== null && $.trim(v.nodeValue) !== '') {
+ if(with_values) { attr[v.nodeName] = v.nodeValue; }
+ else { attr.push(v.nodeName); }
+ }
+ });
+ return attr;
+ };
+})(jQuery);
+
+/*
+Function: $.vakata.get_scrollbar_width
+Gets the width of the scrollbar
+*/
+(function ($) {
+ var sb;
+ $.vakata.get_scrollbar_width = function () {
+ var e1, e2;
+ if(!sb) {
+ if(/msie/.test(navigator.userAgent.toLowerCase())) {
+ e1 = $('<textarea cols="10" rows="2"></textarea>').css({ position: 'absolute', top: -1000, left: 0 }).appendTo('body');
+ e2 = $('<textarea cols="10" rows="2" style="overflow: hidden;"></textarea>').css({ position: 'absolute', top: -1000, left: 0 }).appendTo('body');
+ sb = e1.width() - e2.width();
+ e1.add(e2).remove();
+ }
+ else {
+ e1 = $('<div />').css({ width: 100, height: 100, overflow: 'auto', position: 'absolute', top: -1000, left: 0 })
+ .prependTo('body').append('<div />').find('div').css({ width: '100%', height: 200 });
+ sb = 100 - e1.width();
+ e1.parent().remove();
+ }
+ }
+ return sb;
+ };
+})(jQuery);
+
+/*
+Group: CSS
+Functions needed to manipulate stylesheets (add, remove, change)
+*/
+(function ($) {
+ /*
+ Variable: $.vakata.css
+ *object* holds all CSS related functions
+ */
+ $.vakata.css = {
+ /*
+ Function: $.vakata.css.get_css
+ Retrieves or deletes a specific rule.
+
+ Parameters:
+ rule_name - *string* the rule to search for (any CSS rule)
+ delete_flag - *boolean* whether you want to delete or simply retrieve a reference to the rule
+ sheet - the sheet to search in (do not specify this to search in all sheets)
+
+ Returns either:
+ a reference to the rule - if it was found and the delete flag was not set
+ true - if the delete flag was set and the rule was successfully removed
+ false - if the rule could not be found
+
+ See also:
+ <$.vakata.css.remove_css>
+ */
+ get_css : function(rule_name, delete_flag, sheet) {
+ rule_name = rule_name.toLowerCase();
+ var css_rules = sheet.cssRules || sheet.rules,
+ j = 0;
+ do {
+ if(css_rules.length && j > css_rules.length + 5) { return false; }
+ if(css_rules[j].selectorText && css_rules[j].selectorText.toLowerCase() == rule_name) {
+ if(delete_flag === true) {
+ if(sheet.removeRule) { sheet.removeRule(j); }
+ if(sheet.deleteRule) { sheet.deleteRule(j); }
+ return true;
+ }
+ else { return css_rules[j]; }
+ }
+ }
+ while (css_rules[++j]);
+ return false;
+ },
+ /*
+ Function: $.vakata.css.add_css
+ Adds a rule.
+
+ Parameters:
+ rule_name - *string* the rule to add
+ sheet - a reference to the sheet to add to
+
+ Returns either:
+ a reference to the rule - if the rule was added
+ false - if the rule could not be added, or if such a rule already exists
+ */
+ add_css : function(rule_name, sheet) {
+ if($.jstree.css.get_css(rule_name, false, sheet)) { return false; }
+ if(sheet.insertRule) { sheet.insertRule(rule_name + ' { }', 0); } else { sheet.addRule(rule_name, null, 0); }
+ return $.vakata.css.get_css(rule_name);
+ },
+ /*
+ Function: $.vakata.css.remove_css
+ Removes a rule, this functions is a shortcut to <$.vakata.css.get_css> with the delete flag set to true.
+
+ Parameters:
+ rule_name - *string* the rule to remove
+ sheet - the sheet to remove from (you can omit this and all sheets will be searched)
+
+ Returns either:
+ true - if rule was removed
+ false - if the rule could not be removed
+
+ See also:
+ <$.vakata.css.get_css>
+ */
+ remove_css : function(rule_name, sheet) {
+ return $.vakata.css.get_css(rule_name, true, sheet);
+ },
+ /*
+ Function: $.vakata.css.add_sheet
+ Adds a whole stylesheet or appends to an existing stylesheet.
+
+ Parameters:
+ options - *object*:
+ options.url - location of the stylesheet - when this is supplied _options.str_ and _options.title_ should not be set and a new LINK element is always created
+ options.str - text content of the stylesheet - when this is supplied _options.url_ is not used. A STYLE element is used.
+ options.title - the ID of the added stylesheet (if you pass `foo` the ID will be `foo-stylesheet`), when the stylesheet exists the content is appended and no new stylesheet is created.
+
+ Returns:
+ a reference to the stylesheet
+ */
+ add_sheet : function(opts) {
+ var tmp = false, is_new = true;
+ if(opts.str) {
+ if(opts.title) { tmp = $("style[id='" + opts.title + "-stylesheet']")[0]; }
+ if(tmp) { is_new = false; }
+ else {
+ tmp = document.createElement("style");
+ tmp.setAttribute('type',"text/css");
+ if(opts.title) { tmp.setAttribute("id", opts.title + "-stylesheet"); }
+ }
+ if(tmp.styleSheet) {
+ if(is_new) {
+ document.getElementsByTagName("head")[0].appendChild(tmp);
+ tmp.styleSheet.cssText = opts.str;
+ }
+ else {
+ tmp.styleSheet.cssText = tmp.styleSheet.cssText + " " + opts.str;
+ }
+ }
+ else {
+ tmp.appendChild(document.createTextNode(opts.str));
+ document.getElementsByTagName("head")[0].appendChild(tmp);
+ }
+ return tmp.sheet || tmp.styleSheet;
+ }
+ if(opts.url) {
+ if(document.createStyleSheet) {
+ try { tmp = document.createStyleSheet(opts.url); } catch (e) { }
+ }
+ else {
+ tmp = document.createElement('link');
+ tmp.rel = 'stylesheet';
+ tmp.type = 'text/css';
+ tmp.media = "all";
+ tmp.href = opts.url;
+ document.getElementsByTagName("head")[0].appendChild(tmp);
+ return tmp.styleSheet;
+ }
+ }
+ }
+ };
+})(jQuery);
+
+/*
+Group: Drag'n'drop
+Functions needed to drag'n'drop elements
+*/
+(function ($) {
+ // private variable
+ var vakata_dnd = {
+ element : false,
+ is_down : false,
+ is_drag : false,
+ helper : false,
+ helper_w: 0,
+ data : false,
+ init_x : 0,
+ init_y : 0,
+ scroll_l: 0,
+ scroll_t: 0,
+ scroll_e: false,
+ scroll_i: false
+ };
+ /*
+ Variable: $.vakata.dnd
+ *object* holds all DND related functions
+ */
+ $.vakata.dnd = {
+ /*
+ Variable: $.vakata.dnd.settings
+ *object* holds the global settings object for DND. You can easily modify any of the settings.
+ >// modification example
+ >$.vakata.dnd.settings.threshold = 10;
+ */
+ settings : {
+ /*
+ Variable: $.vakata.dnd.settings.scroll_speed
+ *integer* how fast (pixel count for each step) should a scrollable parent scroll when dragging near the edge. Default is _10_.
+ */
+ scroll_speed : 10,
+ /*
+ Variable: $.vakata.dnd.settings.scroll_proximity
+ *integer* number of pixels from the edge of a scrollable parent below which the parent will start scrolling. Default is _20_.
+ */
+ scroll_proximity : 20,
+ /*
+ Variable: $.vakata.dnd.settings.helper_left
+ *integer* number of pixels left of the cursor to move the drag-helper to. Default is _5_;
+ */
+ helper_left : 5,
+ /*
+ Variable: $.vakata.dnd.settings.helper_top
+ *integer* number of pixels below the cursor to move the drag-helper to. Default is _10_.
+ */
+ helper_top : 10,
+ /*
+ Variable: $.vakata.dnd.settings.threshold
+ *integer* amount of pixels required to move before the drag is started. Default is _5_.
+ */
+ threshold : 5
+ },
+ /*
+ Function: $.vakata.dnd._trigger
+ Used internally to trigger all necessary events.
+ */
+ _trigger : function (event_name, e) {
+ var data = $.vakata.dnd._get();
+ data.event = e;
+ $(document).triggerHandler("dnd_" + event_name + ".vakata", data);
+ },
+ /*
+ Function: $.vakata.dnd._get
+ Used internally to get all items for the drag event. Can be used by foreign code too.
+ */
+ _get : function () {
+ return {
+ "data" : vakata_dnd.data,
+ "element" : vakata_dnd.element,
+ "helper" : vakata_dnd.helper
+ };
+ },
+ /*
+ Function: $.vakata.dnd._clean
+ Used internally to cleanup after a drop, so that all variables are nulled and ready for the next drag.
+ */
+ _clean : function () {
+ if(vakata_dnd.helper) { vakata_dnd.helper.remove(); }
+ if(vakata_dnd.scroll_i) { clearInterval(vakata_dnd.scroll_i); vakata_dnd.scroll_i = false; }
+ vakata_dnd = {
+ element : false,
+ is_down : false,
+ is_drag : false,
+ helper : false,
+ helper_w: 0,
+ data : false,
+ init_x : 0,
+ init_y : 0,
+ scroll_l: 0,
+ scroll_t: 0,
+ scroll_e: false,
+ scroll_i: false
+ };
+ $(document).unbind("mousemove", $.vakata.dnd.drag);
+ $(document).unbind("mouseup", $.vakata.dnd.stop);
+ },
+ /*
+ Function: $.vakata.dnd._scroll
+ Used internally to scroll hovered elements.
+
+ Triggers:
+ <dnd_scroll>
+
+ Event: dnd_scroll
+ Fires when a container is scrolled due to dragging near its edge. Triggered on the document, the event is fired in the *vakata* namespace.
+
+ Parameters:
+ data.event - the scrolled element
+ data.data - the data you supplied when calling <$.vakata.dnd.start>
+ data.element - the origin element
+ data.helper - the jquery extended drag-helper node (or false if it is not used)
+
+ Example:
+ >$(document).bind("dnd_start.vakata", function (e, data) {
+ > // do something
+ >});
+ */
+ _scroll : function (init_only) {
+ if(!vakata_dnd.scroll_e || (!vakata_dnd.scroll_l && !vakata_dnd.scroll_t)) {
+ if(vakata_dnd.scroll_i) { clearInterval(vakata_dnd.scroll_i); vakata_dnd.scroll_i = false; }
+ return false;
+ }
+ if(!vakata_dnd.scroll_i) {
+ vakata_dnd.scroll_i = setInterval($.vakata.dnd._scroll, 100);
+ return false;
+ }
+ if(init_only === true) { return false; }
+
+ var i = vakata_dnd.scroll_e.scrollTop(),
+ j = vakata_dnd.scroll_e.scrollLeft();
+ vakata_dnd.scroll_e.scrollTop(i + vakata_dnd.scroll_t * $.vakata.dnd.settings.scroll_speed);
+ vakata_dnd.scroll_e.scrollLeft(j + vakata_dnd.scroll_l * $.vakata.dnd.settings.scroll_speed);
+ if(i !== vakata_dnd.scroll_e.scrollTop() || j !== vakata_dnd.scroll_e.scrollLeft()) {
+ $.vakata.dnd._trigger("scroll", vakata_dnd.scroll_e);
+ }
+ },
+ /*
+ Function: $.vakata.dnd.start
+ Use this function to start a drag (usually with the mousedown event)
+
+ Parameters:
+ event - *event* the event which started the drag, when used with the mousedown event text selection is prevented
+ data - *mixed* some custom data you want to bind with that particular drag - you will receive this in all events
+ html - *mixed* the text for the drag-helper as a *string*, if set to _false_ no helper is shown
+
+ Returns:
+ false
+
+ Example:
+ >$("span").bind("mousedown", function (e) {
+ > return $.vakata.dnd.start(e, {}, "Dragging");
+ >});
+ */
+ start : function (e, data, html) {
+ if(vakata_dnd.is_drag) { $.vakata.dnd.stop({}); }
+ try {
+ e.currentTarget.unselectable = "on";
+ e.currentTarget.onselectstart = function() { return false; };
+ if(e.currentTarget.style) { e.currentTarget.style.MozUserSelect = "none"; }
+ } catch(err) { }
+ vakata_dnd.init_x = e.pageX;
+ vakata_dnd.init_y = e.pageY;
+ vakata_dnd.data = data;
+ vakata_dnd.is_down = true;
+ vakata_dnd.element = e.currentTarget;
+ if(html !== false) {
+ vakata_dnd.helper = $("<div id='vakata-dnd'></div>").html(html).css({
+ "display" : "block",
+ "margin" : "0",
+ "padding" : "0",
+ "position" : "absolute",
+ "top" : "-2000px",
+ "lineHeight" : "16px",
+ "zIndex" : "10000"
+ });
+ }
+ $(document).bind("mousemove", $.vakata.dnd.drag);
+ $(document).bind("mouseup", $.vakata.dnd.stop);
+ return false;
+ },
+ /*
+ Function: $.vakata.dnd.drag
+ Used internally to process the mousemove event after <$.vakata.dnd.start> is called.
+
+ Parameters:
+ event - *event* the mousemove event
+
+ Triggers:
+ <dnd_start>, <dnd_move>
+ */
+ drag : function (e) {
+ if(!vakata_dnd.is_down) { return; }
+ if(!vakata_dnd.is_drag) {
+ if(
+ Math.abs(e.pageX - vakata_dnd.init_x) > $.vakata.dnd.settings.threshold ||
+ Math.abs(e.pageY - vakata_dnd.init_y) > $.vakata.dnd.settings.threshold
+ ) {
+ if(vakata_dnd.helper) {
+ vakata_dnd.helper.appendTo("body");
+ vakata_dnd.helper_w = vakata_dnd.helper.outerWidth();
+ }
+ vakata_dnd.is_drag = true;
+ /*
+ Event: dnd_start
+ Marks the start of the drag. Triggered on the document after a drag is initiated using <$.vakata.dnd.start> and the user has moved more than <$.vakata.dnd.settings.threshold> pixels, the event is fired in the *vakata* namespace.
+
+ Parameters:
+ data.event - the mousemove event
+ data.data - the data you supplied when calling <$.vakata.dnd.start>
+ data.element - the origin element
+ data.helper - the jquery extended drag-helper node (or false if it is not used)
+
+ Example:
+ >$(document).bind("dnd_start.vakata", function (e, data) {
+ > // do something
+ >});
+ */
+ $.vakata.dnd._trigger("start", e);
+ }
+ else { return; }
+ }
+
+ var d = false, w = false,
+ dh = false, wh = false,
+ dw = false, ww = false,
+ dt = false, dl = false,
+ ht = false, hl = false;
+
+ vakata_dnd.scroll_t = 0;
+ vakata_dnd.scroll_l = 0;
+ vakata_dnd.scroll_e = false;
+ var p = $(e.target)
+ .parentsUntil("body").andSelf().vakata_reverse()
+ .filter(function () {
+ return (/^auto|scroll$/).test($(this).css("overflow")) &&
+ (this.scrollHeight > this.offsetHeight || this.scrollWidth > this.offsetWidth);
+ })
+ .each(function () {
+ var t = $(this), o = t.offset();
+ if(this.scrollHeight > this.offsetHeight) {
+ if(o.top + t.height() - e.pageY < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_t = 1; scr = true; }
+ if(e.pageY - o.top < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_t = -1; scr = true; }
+ }
+ if(this.scrollWidth > this.offsetWidth) {
+ if(o.left + t.width() - e.pageX < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_l = 1; scr = true; }
+ if(e.pageX - o.left < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_l = -1; scr = true; }
+ }
+ if(vakata_dnd.scroll_t || vakata_dnd.scroll_l) {
+ vakata_dnd.scroll_e = $(this);
+ return false;
+ }
+ });
+
+ if(!vakata_dnd.scroll_e) {
+ d = $(document); w = $(window);
+ dh = d.height(); wh = w.height();
+ dw = d.width(); ww = w.width();
+ dt = d.scrollTop(); dl = d.scrollLeft();
+ if(dh > wh && e.pageY - dt < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_t = -1; }
+ if(dh > wh && wh - (e.pageY - dt) < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_t = 1; }
+ if(dw > ww && e.pageX - dl < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_l = -1; }
+ if(dw > ww && ww - (e.pageX - dl) < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_l = 1; }
+ if(vakata_dnd.scroll_t || vakata_dnd.scroll_l) {
+ vakata_dnd.scroll_e = d;
+ }
+ }
+ if(vakata_dnd.scroll_e) { $.vakata.dnd._scroll(true); }
+
+ if(vakata_dnd.helper) {
+ ht = parseInt(e.pageY + $.vakata.dnd.settings.helper_top, 10);
+ hl = parseInt(e.pageX + $.vakata.dnd.settings.helper_left, 10);
+ if(dh && ht + 25 > dh) { ht = dh - 50; }
+ if(dw && hl + vakata_dnd.helper_w > dw) { hl = dw - (vakata_dnd.helper_w + 2); }
+ vakata_dnd.helper.css({
+ left : hl + "px",
+ top : ht + "px"
+ });
+ }
+ /*
+ Event: dnd_move
+ Triggered multiple times while dragging. This event is triggered on the document after the <dnd_start> event when the user moves the mouse, the event is fired in the *vakata* namespace.
+
+ Parameters:
+ data.event - the mousemove event
+ data.data - the data you supplied when calling <$.vakata.dnd.start>
+ data.element - the origin element
+ data.helper - the jquery extended drag-helper node (or false if it is not used)
+
+ Example:
+ >$(document).bind("dnd_move.vakata", function (e, data) {
+ > // do something
+ >});
+ */
+ $.vakata.dnd._trigger("move", e);
+ },
+ /*
+ Function: $.vakata.dnd.stop
+ Used internally to process the mouseup event (drop) after <$.vakata.dnd.start> is called.
+
+ Parameters:
+ event - *event* the mouseup event
+
+ Triggers:
+ <dnd_stop>
+ */
+ stop : function (e) {
+ /*
+ Event: dnd_stop
+ Marks the end of the drag. This event is triggered on the document after <dnd_start> (and possibly <dnd_move>) when a drop (mouseup) occurs or when the drag is programatically terminated, the event is fired in the *vakata* namespace.
+
+ Parameters:
+ data.event - the mouseup event (or _null_ if stopped programatically using <$.vakata.dnd.stop>())
+ data.data - the data you supplied when calling <$.vakata.dnd.start>
+ data.element - the origin element
+ data.helper - the jquery extended drag-helper node (or false if it is not used)
+
+ Example:
+ >$(document).bind("dnd_stop.vakata", function (e, data) {
+ > // do something
+ >});
+ */
+ if(vakata_dnd.is_drag) {
+ $.vakata.dnd._trigger("stop", e);
+ }
+ $.vakata.dnd._clean();
+ }
+ };
+})(jQuery);
+
+/*
+Group: XSLT
+A function used to do XSLT transformations.
+*/
+(function ($) {
+ /*
+ Function: $.vakata.xslt
+ This functions transforms a XML string using a XSL string. The result is passed to a callback function.
+
+ Parameters:
+ xml - *string* the source xml string
+ xsl - *string* the xsl string
+
+ Returns:
+ the transformed result (or _false_ on failure)
+
+ Example:
+ >// simple
+ >$.vakata.xslt("<xml-string-here>", "<xsl-string-here>", function (res) { $("#some-container").append(res); });
+ >// with scope
+ >$.vakata.xslt("<xml-string-here>", "<xsl-string-here>", $.proxy(function (res) {
+ > this.some_process(res);
+ >}, some_object);
+ */
+ $.vakata.xslt = function (xml, xsl) {
+ var r = false, p, q, s;
+ // IE9
+ if(r === false && window.ActiveXObject) {
+ try {
+ r = new ActiveXObject("Msxml2.XSLTemplate");
+ q = new ActiveXObject("Msxml2.DOMDocument");
+ q.loadXML(xml);
+ s = new ActiveXObject("Msxml2.FreeThreadedDOMDocument");
+ s.loadXML(xsl);
+ r.stylesheet = s;
+ p = r.createProcessor();
+ p.input = q;
+ p.transform();
+ r = p.output;
+ }
+ catch (e) { }
+ }
+ xml = $.parseXML(xml);
+ xsl = $.parseXML(xsl);
+ // FF, Chrome
+ if(r === false && typeof (XSLTProcessor) != "undefined") {
+ p = new XSLTProcessor();
+ p.importStylesheet(xsl);
+ r = p.transformToFragment(xml, document);
+ r = $('<div />').append(r).html();
+ }
+ // OLD IE
+ if(r === false && typeof (xml.transformNode) != "undefined") {
+ r = xml.transformNode(xsl);
+ }
+ return r;
+ };
+})(jQuery);
+
+/*
+Group: Context menu
+Functions needed to show a custom context menu.
+*/
+(function ($) {
+ var right_to_left = false,
+ vakata_context = {
+ element : false,
+ reference : false,
+ position_x : 0,
+ position_y : 0,
+ items : [],
+ html : "",
+ is_visible : false
+ };
+ /*
+ Variable: $.vakata.context
+ *object* holds all context menu related functions and variables.
+ */
+ $.vakata.context = {
+ /*
+ Variable: $.vakata.context.settings
+ *object* holds the global settings object for context menus. You can easily modify any of the settings.
+ >// modification example
+ >$.vakata.context.settings.icons = false;
+ */
+ settings : {
+ /*
+ Variable: $.vakata.context.settings.hide_onmouseleave
+ *integer* the amount of milliseconds to wait before hiding the menu after mouseleave. If set to _0_ the menu won't hide on mouseleave. Default is _0_.
+ */
+ hide_onmouseleave : 0,
+ /*
+ Variable: $.vakata.context.settings.icons
+ *boolean* whether to show icons or not. Default is _true_.
+ */
+ icons : true
+ },
+ /*
+ Function: $.vakata.context._trigger
+ Used internally to trigger all necessary events.
+ */
+ _trigger : function (event_name) {
+ $(document).triggerHandler("context_" + event_name + ".vakata", {
+ "reference" : vakata_context.reference,
+ "element" : vakata_context.element,
+ "position" : {
+ "x" : vakata_context.position_x,
+ "y" : vakata_context.position_y
+ }
+ });
+ },
+ /*
+ Function: $.vakata.context._execute
+ Used internally to execute the action (if any) associated with an item.
+
+ Parameters:
+ i - the item's internal index
+ */
+ _execute : function (i) {
+ i = vakata_context.items[i];
+ return i && !i._disabled && i.action ? i.action.call(null, {
+ "item" : i,
+ "reference" : vakata_context.reference,
+ "element" : vakata_context.element,
+ "position" : {
+ "x" : vakata_context.position_x,
+ "y" : vakata_context.position_y
+ }
+ }) : false;
+ },
+ /*
+ Function: $.vakata.context._parse
+ Used internally to parse a contextmenu description object to an HTML string.
+
+ Parameters:
+ o - *object* the contextmenu description object
+ is_callback - *boolean* used internally to indicate a recursive call
+
+ Triggers:
+ <context_parse>
+ */
+ _parse : function (o, is_callback) {
+ if(!o) { return false; }
+ if(!is_callback) {
+ vakata_context.html = "";
+ vakata_context.items = [];
+ }
+ var str = "",
+ sep = false,
+ tmp;
+
+ if(is_callback) { str += "<ul>"; }
+ $.each(o, function (i, val) {
+ if(!val) { return true; }
+ vakata_context.items.push(val);
+ if(!sep && val.separator_before) {
+ str += "<li class='vakata-context-separator'><a href='#' " + ($.vakata.context.settings.icons ? '' : 'style="margin-left:0px;"') + ">&#160;</a></li>";
+ }
+ sep = false;
+ str += "<li class='" + (val._class || "") + (val._disabled ? " vakata-contextmenu-disabled " : "") + "'>";
+ str += "<a href='#' rel='" + (vakata_context.items.length - 1) + "'>";
+ if($.vakata.context.settings.icons) {
+ str += "<ins ";
+ if(val.icon) {
+ if(val.icon.indexOf("/") !== -1) { str += " style='background:url(\"" + val.icon + "\") center center no-repeat' "; }
+ else { str += " class='" + val.icon + "' "; }
+ }
+ str += ">&#160;</ins><span>&#160;</span>";
+ }
+ str += val.label + "</a>";
+ if(val.submenu) {
+ tmp = $.vakata.context._parse(val.submenu, true);
+ if(tmp) { str += tmp; }
+ }
+ str += "</li>";
+ if(val.separator_after) {
+ str += "<li class='vakata-context-separator'><a href='#' " + ($.vakata.context.settings.icons ? '' : 'style="margin-left:0px;"') + ">&#160;</a></li>";
+ sep = true;
+ }
+ });
+ str = str.replace(/<li class\='vakata-context-separator'\><\/li\>$/,"");
+ if(is_callback) { str += "</ul>"; }
+ /*
+ Event: context_parse
+ Triggered when the context menu is parsed but not yet shown. This event is triggered on the document in the *vakata* namespace.
+
+ Parameters:
+ reference - the DOM node used when <$.vakata.context.show> was called
+ element - the DOM node of the context menu (not yet populated and shown)
+ position - an object consisting of _x_ and _y_ keys, represinting the position of the menu (not yet shown)
+
+ Example:
+ >$(document).bind("context_parse.vakata", function (e, data) {
+ > // do something
+ >});
+ */
+ if(!is_callback) { vakata_context.html = str; $.vakata.context._trigger("parse"); }
+ return str.length > 10 ? str : false;
+ },
+ /*
+ Function: $.vakata.context._show_submenu
+ Used internally to show a submenu
+ */
+ _show_submenu : function (o) {
+ o = $(o);
+ if(!o.length || !o.children("ul").length) { return; }
+ var e = o.children("ul"),
+ x = o.offset().left + o.outerWidth(),
+ y = o.offset().top,
+ w = e.width(),
+ h = e.height(),
+ dw = $(document).width(),
+ dh = $(document).height();
+ // може да се спести е една проверка - дали няма някой от класовете вече нагоре
+ if(right_to_left) {
+ o[x - (w + 10 + o.outerWidth()) < 0 ? "addClass" : "removeClass"]("vakata-context-left");
+ }
+ else {
+ o[x + w + 10 > dw ? "addClass" : "removeClass"]("vakata-context-right");
+ }
+ if(y + h + 10 > dh) {
+ e.css("bottom","-1px");
+ }
+ e.show();
+ },
+
+ /*
+ Function: $.vakata.context.show
+ Shows the context menu. Please note that at least one of _reference_ or _position_ should be specified.
+
+ Parameters:
+ reference - *jquery* associate the menu with a DOM element (optional)
+ position - *object* should contain _x_ and _y_ properties, those are the coordinates to show the menu at (optional
+ data - *object* the contextmenu description object. It should consist of keys, each key should be a <context_menu_item>. If not specified the function will search for $(reference).data('vakata_contextmenu') and use that.
+
+ Triggers:
+ <context_show>
+
+ Example:
+ >$(document).bind("contextmenu", function (e) {
+ > e.preventDefault();
+ > $.vakata.context.show(false, { x: e.pageX, y:e.pageY }, {
+ > "create" : {
+ > // only specify what you need
+ > "separator_after" : true,
+ > "label" : "Create",
+ > "action" : function (data) { alert("Create"); }
+ > },
+ > "rename" : {
+ > "label" : "Rename",
+ > "icon" : "./some-icon.png",
+ > "action" : function (data) { alert("Rename on " + data.reference); }
+ > },
+ > "edit" : {
+ > "label" : "Edit",
+ > // Clicking this won't hide the menu, the same can be achieved with:
+ > // "action" : function () { return false; }
+ > "submenu" : {
+ > "copy" : { "label" : "Copy", "action" : function () { } },
+ > "cut" : { "label" : "Cut", "action" : function () { } },
+ > "paste" : { "label" : "Paste", "_disabled" : true, "action" : function () { } }
+ > }
+ > },
+ > "delete" : {
+ > "separator_before" : true,
+ > "label" : "Delete",
+ > "action" : function (data) { alert("Delete"); }
+ > }
+ > });
+ >});
+
+ Variable: context_menu_item
+ *object* Used to construct a context menu entry, this structure will always be a part of an object.
+
+ separator_before - *boolean* should there be a separator before the item. Default is _false_.
+ separator_after - *boolean* should there be a separator after the item. Default is _false_.
+ icon - *string* if supplied this string is used for an icon, if it contains _/_ it is treated as file, otherwise it is applied as a class on an INS object.
+ label - *string* the text for this item
+ submenu - *object* if supplied this object is used to build a submenu. It should consist of keys, each of which is a <context_menu_item>.
+ _class - *string* if supplied this class is applied to the LI node.
+ _disabled - *boolean* is this item disabled.
+ action - *functon* if supplied it will be executed when this item is clicked / activated. If not supplied or the function returns _false_ the contextmenu won't be hidden after execution. To force a context use _$.proxy_.
+ In the function you will receive a single argument which is an object, consisting of four keys:
+ _item_ (the <context_menu_item> object),
+ _reference_ (the DOM node used when <$.vakata.context.show> was called),
+ _element_ (the DOM node of the context menu),
+ _position_ (an object consisting of _x_ and _y_ keys, represinting the current position of the menu)
+
+ See also:
+ <$.vakata.context.show>
+ */
+ show : function (reference, position, data) {
+ if(vakata_context.element && vakata_context.element.length) {
+ vakata_context.element.width('');
+ }
+ switch(!0) {
+ case (!position && !reference):
+ return false;
+ case (!!position && !!reference):
+ vakata_context.reference = reference;
+ vakata_context.position_x = position.x;
+ vakata_context.position_y = position.y;
+ break;
+ case (!position && !!reference):
+ vakata_context.reference = reference;
+ var o = reference.offset();
+ vakata_context.position_x = o.left + reference.outerHeight();
+ vakata_context.position_y = o.top;
+ break;
+ case (!!position && !reference):
+ vakata_context.position_x = position.x;
+ vakata_context.position_y = position.y;
+ break;
+ }
+ if(!!reference && !data && $(reference).data('vakata_contextmenu')) {
+ data = $(reference).data('vakata_contextmenu');
+ }
+ if($.vakata.context._parse(data)) {
+ vakata_context.element.html(vakata_context.html);
+ }
+ if(vakata_context.items.length) {
+ var e = vakata_context.element,
+ x = vakata_context.position_x,
+ y = vakata_context.position_y,
+ w = e.width(),
+ h = e.height(),
+ dw = $(document).width(),
+ dh = $(document).height();
+
+ if(x + w + 20 > dw) {
+ x = dw - (w + 20);
+ }
+ if(y + h + 20 > dh) {
+ y = dh - (h + 20);
+ }
+
+ vakata_context.element
+ .css({ "left" : x, "top" : y })
+ .show()
+ .width(vakata_context.element.outerWidth()); // for ie6
+ vakata_context.is_visible = true;
+ /*
+ Event: context_show
+ Triggered when the context menu is shown. This event is triggered on the document in the *vakata* namespace.
+
+ Parameters:
+ reference - the DOM node used when <$.vakata.context.show> was called
+ element - the DOM node of the context menu
+ position - an object consisting of _x_ and _y_ keys, represinting the position of the menu
+
+ Example:
+ >$(document).bind("context_show.vakata", function (e, data) {
+ > // do something
+ >});
+ */
+ $.vakata.context._trigger("show");
+ }
+ },
+ /*
+ Function: $.vakata.context.hide
+ Used internally to hide the contextmenu after a click, or on mouseleave, etc.
+
+ Triggers:
+ <context_hide>
+ */
+ hide : function () {
+ if(vakata_context.is_visible) {
+ vakata_context.element.hide().find("ul").hide();
+ vakata_context.is_visible = false;
+ /*
+ Event: context_hide
+ Triggered when the context menu is hidden. This event is triggered on the document in the *vakata* namespace.
+
+ Parameters:
+ reference - the DOM node used when <$.vakata.context.show> was called
+ element - the DOM node of the context menu
+ position - an object consisting of _x_ and _y_ keys, represinting the position of the menu
+
+ Example:
+ >$(document).bind("context_hide.vakata", function (e, data) {
+ > // do something
+ >});
+ */
+ $.vakata.context._trigger("hide");
+ }
+ }
+ };
+ $(function () {
+ right_to_left = $("body").css("direction") === "rtl";
+ var to = false,
+ css_string = '' +
+ '.vakata-context { display:none; _width:1px; } ' +
+ '.vakata-context, ' +
+ '.vakata-context ul { margin:0; padding:2px; position:absolute; background:#f5f5f5; border:1px solid #979797; ' +
+ ' -moz-box-shadow:5px 5px 4px -4px #666666; -webkit-box-shadow:2px 2px 2px #999999; box-shadow:2px 2px 2px #999999; }' +
+ '.vakata-context ul { list-style:none; left:100%; margin-top:-2.7em; margin-left:-4px; } ' +
+ '.vakata-context li.vakata-context-right ul { left:auto; right:100%; margin-left:auto; margin-right:-4px; } ' +
+ '.vakata-context li { list-style:none; display:inline; }' +
+ '.vakata-context li a { display:block; padding:0 2em 0 2em; text-decoration:none; width:auto; color:black; white-space:nowrap; line-height:2.4em; ' +
+ ' -moz-text-shadow:1px 1px 0px white; -webkit-text-shadow:1px 1px 0px white; text-shadow:1px 1px 0px white; ' +
+ ' -moz-border-radius:1px; -webkit-border-radius:1px; border-radius:1px; }' +
+ '.vakata-context li a:hover { position:relative; background-color:#e8eff7; ' +
+ ' -moz-box-shadow:0px 0px 2px #0a6aa1; -webkit-box-shadow:0px 0px 2px #0a6aa1; box-shadow:0px 0px 2px #0a6aa1; }' +
+ '.vakata-context li.vakata-context-hover > a { position:relative; background-color:#e8eff7; ' +
+ ' -moz-box-shadow:0px 0px 2px #0a6aa1; -webkit-box-shadow:0px 0px 2px #0a6aa1; box-shadow:0px 0px 2px #0a6aa1; }' +
+ '.vakata-context li a.vakata-context-parent { background-image:url(""); background-position:right center; background-repeat:no-repeat; } ' +
+ '.vakata-context li.vakata-context-separator a, ' +
+ '.vakata-context li.vakata-context-separator a:hover { background:white; border:0; border-top:1px solid #e2e3e3; height:1px; min-height:1px; max-height:1px; padding:0; margin:0 0 0 2.4em; border-left:1px solid #e0e0e0; _overflow:hidden; ' +
+ ' -moz-text-shadow:0 0 0 transparent; -webkit-text-shadow:0 0 0 transparent; text-shadow:0 0 0 transparent; ' +
+ ' -moz-box-shadow:0 0 0 transparent; -webkit-box-shadow:0 0 0 transparent; box-shadow:0 0 0 transparent; ' +
+ ' -moz-border-radius:0; -webkit-border-radius:0; border-radius:0; }' +
+ '.vakata-context li.vakata-contextmenu-disabled a, .vakata-context li.vakata-contextmenu-disabled a:hover { color:silver; background-color:transparent; border:0; box-shadow:0 0 0; }' +
+ '' +
+ '.vakata-context li a ins { text-decoration:none; display:inline-block; width:2.4em; height:2.4em; background:transparent; margin:0 0 0 -2em; } ' +
+ '.vakata-context li a span { display:inline-block; width:1px; height:2.4em; background:white; margin:0 0.5em 0 0; border-left:1px solid #e2e3e3; _overflow:hidden; } ' +
+ '' +
+ '.vakata-context-rtl ul { left:auto; right:100%; margin-left:auto; margin-right:-4px; } ' +
+ '.vakata-context-rtl li a.vakata-context-parent { background-image:url(""); background-position:left center; background-repeat:no-repeat; } ' +
+ '.vakata-context-rtl li.vakata-context-separator a { margin:0 2.4em 0 0; border-left:0; border-right:1px solid #e2e3e3;} ' +
+ '.vakata-context-rtl li.vakata-context-left ul { right:auto; left:100%; margin-left:-4px; margin-right:auto; } ' +
+ '.vakata-context-rtl li a ins { margin:0 -2em 0 0; } ' +
+ '.vakata-context-rtl li a span { margin:0 0 0 0.5em; border-left-color:white; background:#e2e3e3; } ' +
+ '';
+ $.vakata.css.add_sheet({ str : css_string, title : "vakata-context" });
+
+ vakata_context.element = $("<ul class='vakata-context'></ul>");
+ vakata_context.element
+ .delegate("li", "mouseenter", function (e) {
+ e.stopImmediatePropagation();
+
+ if($.contains(this, e.relatedTarget)) {
+ // премахнато заради delegate mouseleave по-долу
+ // $(this).find(".vakata-context-hover").removeClass("vakata-context-hover");
+ return;
+ }
+
+ if(to) { clearTimeout(to); }
+ vakata_context.element.find(".vakata-context-hover").removeClass("vakata-context-hover").end();
+
+ $(this)
+ .siblings().find("ul").hide().end().end()
+ .parentsUntil(".vakata-context", "li").andSelf().addClass("vakata-context-hover");
+ $.vakata.context._show_submenu(this);
+ })
+ // тестово - дали не натоварва?
+ .delegate("li", "mouseleave", function (e) {
+ if($.contains(this, e.relatedTarget)) { return; }
+ $(this).find(".vakata-context-hover").andSelf().removeClass("vakata-context-hover");
+ })
+ .bind("mouseleave", function (e) {
+ $(this).find(".vakata-context-hover").removeClass("vakata-context-hover");
+ if($.vakata.context.settings.hide_onmouseleave) {
+ to = setTimeout(
+ (function (t) {
+ return function () { $.vakata.context.hide(); };
+ })(this), $.vakata.context.settings.hide_onmouseleave);
+ }
+ })
+ .delegate("a", "click", function (e) {
+ e.preventDefault();
+ })
+ .delegate("a", "mouseup", function (e) {
+ if(!$(this).blur().parent().hasClass("vakata-context-disabled") && $.vakata.context._execute($(this).attr("rel")) !== false) {
+ $.vakata.context.hide();
+ }
+ })
+ .appendTo("body");
+
+ $(document)
+ .bind("mousedown", function (e) {
+ if(vakata_context.is_visible && !$.contains(vakata_context.element[0], e.target)) { $.vakata.context.hide(); }
+ })
+ .bind("context_show.vakata", function (e, data) {
+ vakata_context.element.find("li:has(ul)").children("a").addClass("vakata-context-parent");
+ if(right_to_left) {
+ vakata_context.element.addClass("vakata-context-rtl").css("direction", "rtl");
+ }
+ // also apply a RTL class?
+ vakata_context.element.find("ul").hide().end();
+ });
+
+ if(typeof $.hotkeys !== "undefined") {
+ $(document)
+ .bind("keydown", "up", function (e) {
+ if(vakata_context.is_visible) {
+ var o = vakata_context.element.find("ul:visible").andSelf().last().children(".vakata-context-hover").removeClass("vakata-context-hover").prevAll("li:not(.vakata-context-separator)").first();
+ if(!o.length) { o = vakata_context.element.find("ul:visible").andSelf().last().children("li:not(.vakata-context-separator)").last(); }
+ o.addClass("vakata-context-hover");
+ e.stopImmediatePropagation();
+ e.preventDefault();
+ }
+ })
+ .bind("keydown", "down", function (e) {
+ if(vakata_context.is_visible) {
+ var o = vakata_context.element.find("ul:visible").andSelf().last().children(".vakata-context-hover").removeClass("vakata-context-hover").nextAll("li:not(.vakata-context-separator)").first();
+ if(!o.length) { o = vakata_context.element.find("ul:visible").andSelf().last().children("li:not(.vakata-context-separator)").first(); }
+ o.addClass("vakata-context-hover");
+ e.stopImmediatePropagation();
+ e.preventDefault();
+ }
+ })
+ .bind("keydown", "right", function (e) {
+ if(vakata_context.is_visible) {
+ vakata_context.element.find(".vakata-context-hover").last().children("ul").show().children("li:not(.vakata-context-separator)").removeClass("vakata-context-hover").first().addClass("vakata-context-hover");
+ e.stopImmediatePropagation();
+ e.preventDefault();
+ }
+ })
+ .bind("keydown", "left", function (e) {
+ if(vakata_context.is_visible) {
+ vakata_context.element.find(".vakata-context-hover").last().parents("li:eq(0)").find("ul").hide().find(".vakata-context-hover").removeClass("vakata-context-hover");
+ e.stopImmediatePropagation();
+ e.preventDefault();
+ }
+ })
+ .bind("keydown", "esc", function (e) {
+ $.vakata.context.hide();
+ e.preventDefault();
+ })
+ .bind("keydown", "space", function (e) {
+ vakata_context.element.find(".vakata-context-hover").last().children("a").click();
+ e.preventDefault();
+ });
+ }
+ });
+})(jQuery);
+
+/*
+Group: JSON
+Functions needed to encode/decode JSON. Based on the jQuery JSON Plugin.
+*/
+(function ($) {
+ // private function for quoting strings
+ var _quote = function (str) {
+ var escapeable = /["\\\x00-\x1f\x7f-\x9f]/g,
+ meta = { '\b':'\\b','\t':'\\t','\n':'\\n','\f':'\\f','\r':'\\r','"' :'\\"','\\':'\\\\' };
+ if(str.match(escapeable)) {
+ return '"' + str.replace(escapeable, function (a) {
+ var c = _meta[a];
+ if(typeof c === 'string') { return c; }
+ c = a.charCodeAt();
+ return '\\u00' + Math.floor(c / 16).toString(16) + (c % 16).toString(16);
+ }) + '"';
+ }
+ return '"' + str + '"';
+ };
+ /*
+ Variable: $.vakata.json
+ *object* holds all JSON related functions.
+ */
+ $.vakata.json = {
+ /*
+ Function: $.vakata.json.encode
+ A function for encoding data in a JSON notated string.
+
+ Parameters:
+ o - *mixed* the data to be encoded
+
+ Returns:
+ string - the encoded data
+ */
+ encode : function (o) {
+ if (o === null) { return "null"; }
+
+ var tmp = [], i;
+ switch(typeof(o)) {
+ case "undefined":
+ return undefined;
+ case "number":
+ case "boolean":
+ return o + "";
+ case "string":
+ return _quote(o);
+ case "object":
+ if($.isFunction(o.toJSON)) {
+ return $.vakata.json.encode(o.toJSON());
+ }
+ if(o.constructor === Date) {
+ return '"' +
+ o.getUTCFullYear() + '-' +
+ String("0" + (o.getUTCMonth() + 1)).slice(-2) + '-' +
+ String("0" + o.getUTCDate()).slice(-2) + 'T' +
+ String("0" + o.getUTCHours()).slice(-2) + ':' +
+ String("0" + o.getUTCMinutes()).slice(-2) + ':' +
+ String("0" + o.getUTCSeconds()).slice(-2) + '.' +
+ String("00" + o.getUTCMilliseconds()).slice(-3) + 'Z"';
+ }
+ if(o.constructor === Array) {
+ for(i = 0; i < o.length; i++) {
+ tmp.push( $.vakata.json.encode(o[i]) || "null" );
+ }
+ return "[" + tmp.join(",") + "]";
+ }
+
+ $.each(o, function (i, v) {
+ if($.isFunction(v)) { return true; }
+ i = typeof i === "number" ? '"' + i + '"' : _quote(i);
+ v = $.vakata.json.encode(v);
+ tmp.push(i + ":" + v);
+ });
+ return "{" + tmp.join(", ") + "}";
+ }
+ },
+ /*
+ Function: $.vakata.json.decode
+ Exists for consistency and is a simple wrapper for jQuery.parseJSON.
+
+ Parameters:
+ json - the string to be decoded
+
+ Returns:
+ Same as jQuery.parseJSON
+ */
+ decode : function (json) {
+ return $.parseJSON(json);
+ }
+ };
+})(jQuery);
+
+/*
+Group: Cookie
+A copy of the jQuery cookie plugin.
+*/
+(function ($) {
+ /*
+ Function: $.vakata.cookie
+ A function for getting and setting cookies.
+
+ Parameters:
+ Same as the original plugin
+
+ Returns:
+ string - the encoded data
+ */
+ $.vakata.cookie = function (key, value, options) {
+ var days, t, result, decode;
+ if (arguments.length > 1 && String(value) !== "[object Object]") {
+ options = $.extend({}, options);
+ if(value === null || value === undefined) { options.expires = -1; }
+ if(typeof options.expires === 'number') { days = options.expires; t = options.expires = new Date(); t.setDate(t.getDate() + days); }
+ value = String(value);
+ return (document.cookie = [
+ encodeURIComponent(key), '=',
+ options.raw ? value : encodeURIComponent(value),
+ options.expires ? '; expires=' + options.expires.toUTCString() : '',
+ options.path ? '; path=' + options.path : '',
+ options.domain ? '; domain=' + options.domain : '',
+ options.secure ? '; secure' : ''
+ ].join(''));
+ }
+ options = value || {};
+ decode = options.raw ? function (s) { return s; } : decodeURIComponent;
+ return (result = new RegExp('(?:^|; )' + encodeURIComponent(key) + '=([^;]*)').exec(document.cookie)) ? decode(result[1]) : null;
+ };
+})(jQuery);
+
+/*
+Group: LocalStorage
+Functions for dealing with localStorage with fallback to userData or cookies. A slight modification of jstorage.
+*/
+
+(function ($) {
+ var _storage = {},
+ _storage_service = {jStorage:"{}"},
+ _storage_elm = null,
+ _storage_size = 0,
+ json_encode = $.vakata.json.encode,
+ json_decode = $.vakata.json.decode,
+ _backend = false,
+ _ttl_timeout = false;
+
+ function _init() {
+ var localStorageReallyWorks = false;
+ if("localStorage" in window){
+ try {
+ window.localStorage.setItem('_tmptest', 'tmpval');
+ localStorageReallyWorks = true;
+ window.localStorage.removeItem('_tmptest');
+ } catch(BogusQuotaExceededErrorOnIos5) {
+ // Thanks be to iOS5 Private Browsing mode which throws
+ // QUOTA_EXCEEDED_ERRROR DOM Exception 22.
+ }
+ }
+
+ if(localStorageReallyWorks){
+ try {
+ if(window.localStorage) {
+ _storage_service = window.localStorage;
+ _backend = "localStorage";
+ }
+ } catch(E3) {/* Firefox fails when touching localStorage and cookies are disabled */}
+ }
+ else if("globalStorage" in window) {
+ try {
+ if(window.globalStorage) {
+ _storage_service = window.globalStorage[window.location.hostname];
+ _backend = "globalStorage";
+ }
+ } catch(E4) {/* Firefox fails when touching localStorage and cookies are disabled */}
+ }
+ else {
+ _storage_elm = document.createElement('link');
+ if(_storage_elm.addBehavior) {
+ _storage_elm.style.behavior = 'url(#default#userData)';
+ document.getElementsByTagName('head')[0].appendChild(_storage_elm);
+ try {
+ _storage_elm.load("jStorage");
+ var data = "{}";
+ data = _storage_elm.getAttribute("jStorage");
+ _storage_service.jStorage = data;
+ _backend = "userDataBehavior";
+ } catch(E5) {}
+ }
+ if(
+ !_backend && (
+ !!$.vakata.cookie('__vjstorage') ||
+ ($.vakata.cookie('__vjstorage', '{}', { 'expires' : 365 }) && $.vakata.cookie('__vjstorage') === '{}')
+ )
+ ) {
+ _storage_elm = null;
+ _storage_service.jStorage = $.vakata.cookie('__vjstorage');
+ _backend = "cookie";
+ }
+
+ if(!_backend) {
+ _storage_elm = null;
+ return;
+ }
+ }
+ _load_storage();
+ _handleTTL();
+ }
+
+ function _load_storage() {
+ if(_storage_service.jStorage) {
+ try {
+ _storage = json_decode(String(_storage_service.jStorage));
+ } catch(E6) { _storage_service.jStorage = "{}"; }
+ } else {
+ _storage_service.jStorage = "{}";
+ }
+ _storage_size = _storage_service.jStorage ? String(_storage_service.jStorage).length : 0;
+ }
+
+ function _save() {
+ try {
+ _storage_service.jStorage = json_encode(_storage);
+ if(_backend === 'userDataBehavior') {
+ _storage_elm.setAttribute("jStorage", _storage_service.jStorage);
+ _storage_elm.save("jStorage");
+ }
+ if(_backend === 'cookie') {
+ $.vakata.cookie('__vjstorage', _storage_service.jStorage, { 'expires' : 365 });
+ }
+ _storage_size = _storage_service.jStorage?String(_storage_service.jStorage).length:0;
+ } catch(E7) { /* probably cache is full, nothing is saved this way*/ }
+ }
+
+ function _checkKey(key) {
+ if(!key || (typeof key != "string" && typeof key != "number")){
+ throw new TypeError('Key name must be string or numeric');
+ }
+ if(key == "__jstorage_meta") {
+ throw new TypeError('Reserved key name');
+ }
+ return true;
+ }
+
+ function _handleTTL() {
+ var curtime = +new Date(),
+ i,
+ TTL,
+ nextExpire = Infinity,
+ changed = false;
+
+ if(_ttl_timeout !== false) {
+ clearTimeout(_ttl_timeout);
+ }
+ if(!_storage.__jstorage_meta || typeof _storage.__jstorage_meta.TTL != "object"){
+ return;
+ }
+ TTL = _storage.__jstorage_meta.TTL;
+ for(i in TTL) {
+ if(TTL.hasOwnProperty(i)) {
+ if(TTL[i] <= curtime) {
+ delete TTL[i];
+ delete _storage[i];
+ changed = true;
+ }
+ else if(TTL[i] < nextExpire) {
+ nextExpire = TTL[i];
+ }
+ }
+ }
+
+ // set next check
+ if(nextExpire != Infinity) {
+ _ttl_timeout = setTimeout(_handleTTL, nextExpire - curtime);
+ }
+ // save changes
+ if(changed) {
+ _save();
+ }
+ }
+
+ /*
+ Variable: $.vakata.storage
+ *object* holds all storage related functions and properties.
+ */
+ $.vakata.storage = {
+ /*
+ Variable: $.vakata.storage.version
+ *string* the version of jstorage used
+ */
+ version: "0.1.6.1",
+ /*
+ Function: $.vakata.storage.set
+ Set a key to a value
+
+ Parameters:
+ key - the key
+ value - the value
+
+ Returns:
+ _value_
+ */
+ set : function (key, value) {
+ _checkKey(key);
+ _storage[key] = value;
+ _save();
+ return value;
+ },
+ /*
+ Function: $.vakata.storage.get
+ Get a value by key.
+
+ Parameters:
+ key - the key
+ def - the value to return if _key_ is not found
+
+ Returns:
+ The found value, _def_ if key not found or _null_ if _def_ is not supplied.
+ */
+ get : function (key, def) {
+ _checkKey(key);
+ if(key in _storage){
+ return _storage[key];
+ }
+ return typeof(def) == 'undefined' ? null : def;
+ },
+ /*
+ Function: $.vakata.storage.del
+ Remove a key.
+
+ Parameters:
+ key - the key
+
+ Returns:
+ *boolean*
+ */
+ del : function (key) {
+ _checkKey(key);
+ if(key in _storage) {
+ delete _storage[key];
+
+ if(_storage.__jstorage_meta && typeof _storage.__jstorage_meta.TTL == "object" && key in _storage.__jstorage_meta.TTL) {
+ delete _storage.__jstorage_meta.TTL[key];
+ }
+ _save();
+ return true;
+ }
+ return false;
+ },
+
+ setTTL: function(key, ttl){
+ var curtime = +new Date();
+
+ _checkKey(key);
+ ttl = Number(ttl) || 0;
+ if(key in _storage){
+ if(!_storage.__jstorage_meta){
+ _storage.__jstorage_meta = {};
+ }
+ if(!_storage.__jstorage_meta.TTL) {
+ _storage.__jstorage_meta.TTL = {};
+ }
+ if(ttl > 0) {
+ _storage.__jstorage_meta.TTL[key] = curtime + ttl;
+ }
+ else {
+ delete _storage.__jstorage_meta.TTL[key];
+ }
+ _save();
+ _handleTTL();
+ return true;
+ }
+ return false;
+ },
+
+ /*
+ Function: $.vakata.storage.flush
+ Empty the storage.
+
+ Returns:
+ _true_
+ */
+ flush : function(){
+ _storage = {};
+ _save();
+ // try{ window.localStorage.clear(); } catch(E8) { }
+ return true;
+ },
+ /*
+ Function: $.vakata.storage.storageObj
+ Get a read only copy of the whole storage.
+
+ Returns:
+ *object*
+ */
+ storageObj : function(){
+ function F() {}
+ F.prototype = _storage;
+ return new F();
+ },
+ /*
+ Function: $.vakata.storage.index
+ Get an array of all the set keys in the storage.
+
+ Returns:
+ *array*
+ */
+ index : function(){
+ var index = [], i;
+ $.each(_storage, function (i, v) { if(i != "__jstorage_meta") { index.push(i); } });
+ return index;
+ },
+ /*
+ Function: $.vakata.storage.storageSize
+ Get the size of all items in the storage in bytes.
+
+ Returns:
+ *number*
+ */
+ storageSize : function(){
+ return _storage_size;
+ },
+ /*
+ Function: $.vakata.storage.currentBackend
+ Get the current backend used.
+
+ Returns:
+ *string*
+ */
+ currentBackend : function(){
+ return _backend;
+ },
+ /*
+ Function: $.vakata.storage.storageAvailable
+ See if storage functionality is available.
+
+ Returns:
+ *boolean*
+ */
+ storageAvailable : function(){
+ return !!_backend;
+ }
+ };
+ _init();
+})(jQuery);
+
+/*
+Group: PrettyDate
+Modifies time elements to a more human readable value. Taken from: https://github.com/zachleat/Humane-Dates/blob/master/src/humane.js
+*/
+(function ($) {
+ /*
+ Variable: $.vakata.pretty_date
+ *object* holds all pretty-date related functions and properties.
+ */
+ $.vakata.pretty_date = {
+ /*
+ Variable: $.vakata.pretty_date.lang
+ *object* the localization to use.
+ */
+ lang : {
+ ago: 'Ago',
+ from: 'From Now',
+ now: 'Just Now',
+ minute: 'Minute',
+ minutes: 'Minutes',
+ hour: 'Hour',
+ hours: 'Hours',
+ day: 'Day',
+ days: 'Days',
+ week: 'Week',
+ weeks: 'Weeks',
+ month: 'Month',
+ months: 'Months',
+ year: 'Year',
+ years: 'Years'
+ },
+ /*
+ Function: $.vakata.pretty_date.parse
+ Parses the difference between to dates to a human readable string.
+
+ Parameters:
+ date - the date to calculate from (а string in this YYYY-MM-DDTHH:MM:SSZ format - UTC)
+ comareTo - the date to compare to (as date), if left empty the current date is used
+
+ Returns:
+ *mixed* - the formatted string on success or _null_ on error
+ */
+ parse : function (date, compareTo) {
+ // remove the timezone (always use gmdate on server side)
+ date = new Date(date.replace(/-/g,"/").replace(/[TZ]/g," ").replace(/\+\d\d\:\d\d$/,''));
+ compareTo = compareTo || new Date();
+ var lang = $.vakata.pretty_date.lang,
+ formats = [
+ [60, lang.now],
+ [3600, lang.minute, lang.minutes, 60], // 60 minutes, 1 minute
+ [86400, lang.hour, lang.hours, 3600], // 24 hours, 1 hour
+ [604800, lang.day, lang.days, 86400], // 7 days, 1 day
+ [2628000, lang.week, lang.weeks, 604800], // ~1 month, 1 week
+ [31536000, lang.month, lang.months, 2628000], // 1 year, ~1 month
+ [Infinity, lang.year, lang.years, 31536000] // Infinity, 1 year
+ ],
+ seconds = (compareTo - date + compareTo.getTimezoneOffset() * 60000) / 1000,
+ normalize = function (val, single) {
+ var margin = 0.1;
+ if(val >= single && val <= single * (1+margin)) {
+ return single;
+ }
+ return val;
+ },
+ token;
+
+ if(seconds < 0) {
+ seconds = Math.abs(seconds);
+ token = ' ' + lang.from;
+ }
+ else {
+ token = ' ' + lang.ago;
+ }
+
+ for(var i = 0, format = formats[0]; formats[i]; format = formats[++i]) {
+ if(seconds < format[0]) {
+ if(i === 0) {
+ return format[1];
+ }
+ var val = Math.ceil(normalize(seconds, format[3]) / (format[3]));
+ return val +
+ ' ' +
+ (val != 1 ? format[2] : format[1]) +
+ (i > 0 ? token : '');
+ }
+ }
+ },
+ /*
+ Function: $.vakata.pretty_date.init
+ Parses all time elements in the document and keeps reparsing them every few seconds.
+
+ Parameters:
+ i - the interval for reparsing (in milliseconds). Default is 60000.
+ format - the format to use, example: _Published %{s}._. Default is _%{s}_.
+ */
+ init : function (i, format) {
+ $("time, [datetime]").vakata_pretty_date(format);
+ setInterval(function(){ $("time, [datetime]").vakata_pretty_date(format); }, i || 60000);
+ }
+ };
+ /*
+ Function: $().vakata_pretty_date
+ Sets the HTML of every element to the parsed difference of its _datetime_ attribute and the compare parameter.
+
+ Parameters:
+ format - makes it possible to modify the parsed string, example: _Published %{s}._. Default is _%{s}_.
+ compare - the date to compare to. Default is the current date.
+ */
+ $.fn.vakata_pretty_date = function (format, compare) {
+ if(!format) { format = '%{s}'; }
+ return this.each(function() {
+ var $t = jQuery(this),
+ date = $.vakata.pretty_date.parse($t.attr('datetime'), compare);
+ if(date) {
+ date = format.replace('%{s}', date);
+ if($t.html() != date) {
+ $t.html(date);
+ }
+ }
+ });
+ };
+})(jQuery);
+
+/*
+Group: Selection
+Selection related functions
+*/
+(function ($) {
+ /*
+ Variable: $.vakata.selection
+ *object* holds all selection related functions and properties.
+ */
+ $.vakata.selection = {
+ /*
+ Function: $.vakata.selection.get
+ Gets the current selection.
+
+ Parameters:
+ as_text - a boolean - if set to _true_ selection is returned as text, otherwise as HTML
+
+ Returns:
+ *string* - the current selection
+ */
+ get : function (as_text) {
+ if(window.getSelection) {
+ if(as_text) {
+ return window.getSelection().toString();
+ }
+ var userSelection = window.getSelection(),
+ range = userSelection.getRangeAt && userSelection.rangeCount ? userSelection.getRangeAt(0) : document.createRange(),
+ div = document.createElement('div');
+ if(!userSelection.getRangeAt) {
+ range.setStart(userSelection.anchorNode, userSelection.anchorOffset);
+ range.setEnd(userSelection.focusNode, userSelection.focusOffset);
+ }
+ div.appendChild(range.cloneContents());
+ return div.innerHTML;
+ }
+ if(document.selection) {
+ return document.selection.createRange()[ as_text ? 'text' : 'htmlText' ];
+ }
+ return '';
+ },
+ /*
+ Function: $.vakata.selection.elm_get
+ Gets the selection inside an input element or textarea.
+
+ Parameters:
+ e - the actual DOM element or the ID of the element
+
+ Returns:
+ *object* - the current selection (start, end, length, text)
+ */
+ elm_get : function (e) {
+ e = typeof e === 'string' ? document.getElementById(e) : e;
+ if(e.jquery) { e = e.get(0); }
+ if('selectionStart' in e) { // Mozilla and DOM 3.0
+ return {
+ 'start' : e.selectionStart,
+ 'end' : e.selectionEnd,
+ 'length' : (e.selectionEnd - e.selectionStart),
+ 'text' : e.value.substr(e.selectionStart, (e.selectionEnd - e.selectionStart))
+ };
+ }
+ else if(document.selection) { // IE
+ e.focus();
+ var tr0 = document.selection.createRange(),
+ tr1 = false,
+ tr2 = false,
+ len, text_whole, the_start, the_end;
+ if(tr0 && tr0.parentElement() == e) {
+ len = e.value.length;
+ text_whole = e.value.replace(/\r\n/g, "\n");
+
+ tr1 = e.createTextRange();
+ tr1.moveToBookmark(tr0.getBookmark());
+ tr2 = e.createTextRange();
+ tr2.collapse(false);
+
+ if(tr1.compareEndPoints("StartToEnd", tr2) > -1) {
+ the_start = the_end = len;
+ }
+ else {
+ the_start = -tr1.moveStart("character", -len);
+ the_start += text_whole.slice(0, the_start).split("\n").length - 1;
+ if (tr1.compareEndPoints("EndToEnd", tr2) > -1) {
+ the_end = len;
+ } else {
+ the_end = -tr1.moveEnd("character", -len);
+ the_end += text_whole.slice(0, the_end).split("\n").length - 1;
+ }
+ }
+ text_whole = e.value.slice(the_start, the_end);
+ return {
+ 'start' : the_start,
+ 'end' : the_end,
+ 'length' : text_whole.length,
+ 'text' : text_whole
+ };
+ }
+ }
+ else { // not supported
+ return {
+ 'start' : e.value.length,
+ 'end' : e.value.length,
+ 'length' : 0,
+ 'text' : ''
+ };
+ }
+ },
+ /*
+ Function: $.vakata.selection.elm_set
+ Sets the selection inside an input element or textarea.
+
+ Parameters:
+ e - the actual DOM element or the ID of the element
+ beg - the char to start the selection
+ end - the char to end the selection
+
+ Returns:
+ *object* - the current selection (start, end, length, text)
+ */
+ elm_set : function (e, beg, end) {
+ e = typeof e === 'string' ? document.getElementById(e) : e;
+ if(e.jquery) { e = e.get(0); }
+ if('selectionStart' in e) { // Mozilla and DOM 3.0
+ e.focus();
+ e.selectionStart = beg;
+ e.selectionEnd = end;
+ }
+ else if(document.selection) { // IE
+ e.focus();
+ var tr = e.createTextRange(),
+ tx = e.value.replace(/\r\n/g, "\n");
+
+ beg -= tx.slice(0, beg).split("\n").length - 1;
+ end -= tx.slice(0, end).split("\n").length - 1;
+
+ tr.collapse(true);
+ tr.moveEnd('character', end);
+ tr.moveStart('character', beg);
+ tr.select();
+ }
+ return $.vakata.selection.elm_get(e);
+ },
+ /*
+ Function: $.vakata.selection.elm_replace
+ Replace the selection inside an input element or textarea.
+
+ Parameters:
+ e - the actual DOM element or the ID of the element
+ replace - the string to replace the selection with
+
+ Returns:
+ *object* - the current selection (start, end, length, text)
+ */
+ elm_replace : function (e, replace) {
+ e = typeof e === 'string' ? document.getElementById(e) : e;
+ if(e.jquery) { e = e.get(0); }
+ var sel = $.vakata.selection.elm_get(e),
+ beg = sel.start,
+ end = beg + replace.length;
+ elm.value = elm.value.substr(0, beg) + replace + elm.value.substr(sel.end, elm.value.length);
+ $.vakata.selection.elm_set(e, beg, end);
+ return {
+ 'start' : beg,
+ 'end' : end,
+ 'length' : replace.length,
+ 'text' : replace
+ };
+ },
+ /*
+ Function: $.vakata.selection.elm_get_caret
+ Returns the caret position in the element.
+
+ Parameters:
+ e - the actual DOM element or the ID of the element
+
+ Returns:
+ *number* - the current caret position
+ */
+ elm_get_caret : function (e) {
+ return $.vakata.selection.elm_get(e).end;
+ },
+ /*
+ Function: $.vakata.selection.elm_set_caret
+ Sets the caret position in the element.
+
+ Parameters:
+ e - the actual DOM element or the ID of the element
+ pos - the position to move the caret to
+
+ Returns:
+ *object* - the current selection
+ */
+ elm_set_caret : function (e, pos) {
+ return $.vakata.selection.elm_set(e, pos, pos);
+ },
+ /*
+ Function: $.vakata.selection.elm_get_caret_position
+ Returns the caret position in pixels relative to the element.
+
+ Parameters:
+ e - the actual DOM element or the ID of the element
+
+ Returns:
+ *object* - the current position (with _left_ and _top_ values)
+ */
+ elm_get_caret_position : function (e) {
+ e = typeof e === 'string' ? document.getElementById(e) : e;
+ if(e.jquery) { e = e.get(0); }
+ var p = $.vakata.selection.elm_get_caret(e),
+ s = e.value.substring(0, p).replace(/&/g,'&amp;').replace(/</ig,'&lt;').replace(/>/ig,'&gt;').replace(/\r/g, '').replace(/\t/g,'&#10;').replace(/\n/ig, '<br />'),
+ b = $.vakata.get_scrollbar_width(),
+ w = $(e).width(),
+ h = $(e).height();
+ if(e.scrollHeight > h) { w -= b; }
+ if(e.scrollWidth > w) { h -= b; }
+ e = $(e);
+ e = $('<div />').html(s).css({
+ 'background': 'red',
+ 'width' : w + 'px',
+ 'height' : 'auto',
+ 'position' : 'absolute',
+ 'left' : '0px',
+ 'top' : '-10000px',
+
+ 'fontSize' : e.css('fontSize'),
+ 'fontFamily' : e.css('fontFamily'),
+ 'fontWeight' : e.css('fontWeight'),
+ 'fontVariant' : e.css('fontVariant'),
+ 'fontStyle' : e.css('fontStyle'),
+ 'textTransform' : e.css('textTransform'),
+ 'lineHeight' : e.css('lineHeight'),
+ 'whiteSpace' : 'pre-wrap'
+ });
+ e.append('<span class="caret">&nbsp;</span>').appendTo('body');
+ s = e.find('span.caret');
+ p = s.offset();
+ p.top = p.top + 10000 + s.height();
+ e.remove();
+ return p;
+ }
+ };
+})(jQuery); \ No newline at end of file
diff --git a/pom.xml b/pom.xml
index a55230c..5593615 100644
--- a/pom.xml
+++ b/pom.xml
@@ -10,7 +10,7 @@
<url>https://code.renci.org/gf/project/irodsidrop/</url>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
- <jargon.version>3.1.1-SNAPSHOT</jargon.version>
+ <jargon.version>3.1.1</jargon.version>
<spring.core.version>3.0.5.RELEASE</spring.core.version>
<commons.io.version>2.0.1</commons.io.version>
<netbeans.version>RELEASE701</netbeans.version>
diff --git a/release_notes.txt b/release_notes.txt
new file mode 100644
index 0000000..1a92814
--- /dev/null
+++ b/release_notes.txt
@@ -0,0 +1,40 @@
+*'''Project''': iDrop Suite - iDrop Clients
+*'''Date''': 4/13/2012
+*'''Release Version''': 1.0.0
+*'''git tag''': 1.0.0
+
+==News==
+
+This is an initial release of the iDrop Suite components. This serves as an initial checkpoint, and will be followed on by regular updates.
+
+Please go to [[https://code.renci.org/gf/project/irodsidrop/]] for the latest news and info.
+
+iDrop consists of the following libraries
+
+* idrop-swing - Desktop transfer and synchronization manager
+* idrop-lite - Applet plug-in for bulk uploads and downloads
+* idrop-web - Web browser interface for file operations
+
+Other modules are planned in later releases for REST-ful API and mobile access.
+
+==Requirements==
+
+*Jargon depends on Java 1.6+
+*Jargon is built using Apache Maven2, see POM for dependencies
+*Jargon supports iRODS 2.4 through iRODS 3.0, and is tested with the current (as of release time) development version of iRODS
+
+==Libraries==
+
+
+Jargon-core uses Maven for dependency management. See the pom.xml file for references to various dependencies.
+
+Note that the following bug and feature requests are logged in GForge with related commit information [[https://code.renci.org/gf/project/irodsidrop/tracker/]]
+
+
+==Bug Fixes==
+
+
+==Features==
+
+
+ \ No newline at end of file