diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor.xml
new file mode 100644
index 0000000000000000000000000000000000000000..b68162a656d9429574d0dd1eb4b32078a239771d
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor.xml
@@ -0,0 +1,26 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="Folder" module="OFS.Folder"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>_objects</string> </key>
+            <value>
+              <tuple/>
+            </value>
+        </item>
+        <item>
+            <key> <string>id</string> </key>
+            <value> <string>erp5_jexcel_editor</string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string>ERP5 Jexcel Editor</string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/WebSection_getJexcelEditorPrecacheManifestList.py b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/WebSection_getJexcelEditorPrecacheManifestList.py
new file mode 100644
index 0000000000000000000000000000000000000000..5958013d197329a30c97e4d2ffa70c14e89f38f2
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/WebSection_getJexcelEditorPrecacheManifestList.py
@@ -0,0 +1,18 @@
+url_list = [
+  "renderjs.js",
+  "rsvp.js",
+  "jio.js",
+  "domsugar.js",
+  "jsuites/jsuites.js",
+  "jsuites/jsuites.css",
+  "jexcel/jexcel.js",
+  "jexcel/jexcel.css",
+  "jexcel.gadget.html",
+  "jexcel.gadget.js",
+  "complements.css",
+  "fx.png",
+  "fonts.ttf",
+  "icons.css",
+]
+
+return url_list
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/WebSection_getJexcelEditorPrecacheManifestList.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/WebSection_getJexcelEditorPrecacheManifestList.xml
new file mode 100644
index 0000000000000000000000000000000000000000..9e68c64e0929b8ea9d9a829d061bf3e5aa26fc89
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/WebSection_getJexcelEditorPrecacheManifestList.xml
@@ -0,0 +1,62 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="PythonScript" module="Products.PythonScripts.PythonScript"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>Script_magic</string> </key>
+            <value> <int>3</int> </value>
+        </item>
+        <item>
+            <key> <string>_bind_names</string> </key>
+            <value>
+              <object>
+                <klass>
+                  <global name="NameAssignments" module="Shared.DC.Scripts.Bindings"/>
+                </klass>
+                <tuple/>
+                <state>
+                  <dictionary>
+                    <item>
+                        <key> <string>_asgns</string> </key>
+                        <value>
+                          <dictionary>
+                            <item>
+                                <key> <string>name_container</string> </key>
+                                <value> <string>container</string> </value>
+                            </item>
+                            <item>
+                                <key> <string>name_context</string> </key>
+                                <value> <string>context</string> </value>
+                            </item>
+                            <item>
+                                <key> <string>name_m_self</string> </key>
+                                <value> <string>script</string> </value>
+                            </item>
+                            <item>
+                                <key> <string>name_subpath</string> </key>
+                                <value> <string>traverse_subpath</string> </value>
+                            </item>
+                          </dictionary>
+                        </value>
+                    </item>
+                  </dictionary>
+                </state>
+              </object>
+            </value>
+        </item>
+        <item>
+            <key> <string>_params</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>id</string> </key>
+            <value> <string>WebSection_getJexcelEditorPrecacheManifestList</string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/complements.css.css b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/complements.css.css
new file mode 100644
index 0000000000000000000000000000000000000000..e97708a0c2511c489772796f249ccba8c84b56cb
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/complements.css.css
@@ -0,0 +1,84 @@
+div.jexcel_formula img{
+  display: inline-block;
+  width: 12px;
+  height: 12px;
+  margin-left: 4px;
+  margin-right: 4px;
+}
+
+input.jexcel_formula {
+  margin-left: 2px;
+  width: calc(100% - 77px);
+  position:sticky;
+  top:0px;
+  z-index:0;
+  display: inline-block;
+}
+
+div.jexcel-filter > * {
+  display: inline-block;
+}
+
+input.cell_input {
+  display: inline-block;
+  width: 35px;
+}
+
+input.jexcel_search {
+  width: 40px;
+}
+
+.jexcel_toolbar select.minimize {
+  width: 90px;
+}
+
+td {
+  cursor: cell;
+}
+
+div.jexcel_toolbar div.jcolor-content td {
+  cursor: crosshair;
+}
+
+input.readonly {
+  cursor: not-allowed;
+}
+
+body > div.spreadsheet.jexcel_tabs > div:nth-child(1) {
+  position : absolute;
+  bottom: 0;
+}
+
+.add_delete {
+  position : absolute;
+  bottom : 3px;
+  right : 10px;
+}
+
+.add_delete > i {
+  cursor : pointer;
+  margin-left : 10px;
+}
+
+.jexcel_tab.jexcel_container.fullscreen.with-toolbar {
+  position: absolute;
+  top: 0;
+}
+
+.jexcel_tabs .jexcel_tab_link, .jexcel_filter > div {
+  font-family: arial;
+}
+
+.jexcel_tab_link.selected {
+  cursor: alias;
+}
+
+:focus {
+  outline: none;
+}
+
+@media (max-width: 1150px){
+  div.jexcel_filter {
+    display: none;
+  }
+}
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/complements.css.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/complements.css.xml
new file mode 100644
index 0000000000000000000000000000000000000000..559b190f9f5b123ca1c3da7d95590b9a4e691d07
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/complements.css.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>complements.css</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>text/css</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fonts.ttf.ttf b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fonts.ttf.ttf
new file mode 100644
index 0000000000000000000000000000000000000000..8f07978740cecc6083137cb36262a096f6a7df76
Binary files /dev/null and b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fonts.ttf.ttf differ
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fonts.ttf.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fonts.ttf.xml
new file mode 100644
index 0000000000000000000000000000000000000000..d57ff36d118c7f14d7885edd5385d23f3b66c7a1
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fonts.ttf.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>fonts.ttf</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>font/ttf</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fx.png.png b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fx.png.png
new file mode 100644
index 0000000000000000000000000000000000000000..9c2a02f4d84044523ad791c8daf956874c0628af
Binary files /dev/null and b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fx.png.png differ
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fx.png.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fx.png.xml
new file mode 100644
index 0000000000000000000000000000000000000000..46922e690078c390c5c4ce47bcee9b353863ce8c
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/fx.png.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>fx.png</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>image/png</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/icons.css.css b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/icons.css.css
new file mode 100644
index 0000000000000000000000000000000000000000..4dd83990251293279e9a4f1ffcdf10b47d3ca6b9
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/icons.css.css
@@ -0,0 +1,20 @@
+@font-face {
+  font-family: 'Material Icons';
+  font-style: normal;
+  font-weight: 400;
+  src: url(fonts.ttf) format('truetype');
+}
+
+.material-icons {
+  font-family: 'Material Icons';
+  font-weight: normal;
+  font-style: normal;
+  font-size: 24px;
+  line-height: 1;
+  letter-spacing: normal;
+  text-transform: none;
+  display: inline-block;
+  white-space: nowrap;
+  word-wrap: normal;
+  direction: ltr;
+}
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/icons.css.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/icons.css.xml
new file mode 100644
index 0000000000000000000000000000000000000000..933a79037d5cb69700d45dd89c0ca237322e59cf
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/icons.css.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>icons.css</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>text/css</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.html.html b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.html.html
new file mode 100644
index 0000000000000000000000000000000000000000..e27a252c8279f2900371f071957dcaa18c0fa055
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.html.html
@@ -0,0 +1,28 @@
+<!DOCTYPE html>
+<html>
+  <head>
+    <meta charset="utf-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1">
+    <title>Jexcel</title>
+
+    <script src="rsvp.js"></script>
+    <script src="renderjs.js"></script>
+    <script src="jio.js"></script>
+
+    <script src="domsugar.js"></script>
+
+    <script src="jexcel/jexcel.js"></script>
+    <script src="jsuites/jsuites.js"></script>
+    <link rel="stylesheet" href="jexcel/jexcel.css" type="text/css" />
+    <link rel="stylesheet" href="jsuites/jsuites.css" type="text/css" />
+    <link rel="stylesheet" href="complements.css" type="text/css" />
+
+    <link rel="stylesheet" href="icons.css" type="text/css" />
+
+    <script src="jexcel.gadget.js"></script>
+
+  </head>
+  <body>
+    <div class="spreadsheet"></div>
+  </body>
+</html>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.html.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.html.xml
new file mode 100644
index 0000000000000000000000000000000000000000..fb1b432a9a5a69306470dd7a9975ff36ff29fd72
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.html.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>jexcel.gadget.html</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>text/html</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.js.js b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.js.js
new file mode 100644
index 0000000000000000000000000000000000000000..701f4d81b247dd0a4ceb5f84944d2f53ea9a6772
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.js.js
@@ -0,0 +1,1122 @@
+/*jslint nomen: true, indent: 2, maxlen: 80, unparam: true */
+/*global window, rJS, RSVP, jexcel, domsugar, document, alert,
+prompt, confirm, navigator*/
+(function (window, rJS, jexcel, domsugar, document, alert,
+            prompt, confirm, navigator) {
+  "use strict";
+
+  function isMobileDevice() {
+    return (window.orientation !== undefined) ||
+      (navigator.userAgent.indexOf('IEMobile') !== -1);
+  }
+
+  function format(node, level) {
+    var indentBefore = new Array(level + 2).join('  '),
+      indentAfter,
+      textNode,
+      i;
+    level += 1;
+    indentAfter = new Array(level - 1).join('  ');
+    for (i = 0; i < node.children.length; i += 1) {
+      textNode = document.createTextNode('\n' + indentBefore);
+      node.insertBefore(textNode, node.children[i]);
+      format(node.children[i], level);
+      if (node.lastElementChild === node.children[i]) {
+        textNode = document.createTextNode('\n' + indentAfter);
+        node.appendChild(textNode);
+      }
+    }
+    return node;
+  }
+
+  function beautifyHTMLString(str) {
+    var div = domsugar("div", {html: str.trim()});
+    return format(div, 0).innerHTML;
+  }
+
+  function createElementFromHTML(html_string) {
+    var div = domsugar("div", {html: html_string.trim()});
+    return div.children;
+  }
+
+  function columnLetterToNumber(str) {
+    // convert a column letter to its corresponding column number :
+    // A -> 0, B -> 1, AA -> 26, AB -> 27 ...
+    var out = 0, len = str.length, pos = len;
+    while (--pos > -1) {
+      out += (str.charCodeAt(pos) - 64) * Math.pow(26, len - 1 - pos);
+    }
+    return out - 1;
+  }
+
+  function numberToColumnLetter(i) {
+    // convert a column number to its corresponding column letter :
+    // 0 -> A, 1 -> B, 26 -> AA, 27 -> AB ...
+    return (i >= 26 ? numberToColumnLetter((i / 26 >> 0) - 1) : '') +
+      'ABCDEFGHIJKLMNOPQRSTUVWXYZ'[i % 26 >> 0];
+  }
+
+  function getCoordinatesFromCell(cell) {
+    var x = Number(cell.dataset.x),
+      y = Number(cell.dataset.y) + 1;
+    return numberToColumnLetter(x) + y.toString();
+  }
+
+  function fireDoubleClick(element) {
+    var clickEvent  = document.createEvent('MouseEvents');
+    clickEvent.initEvent('dblclick', true, true);
+    element.dispatchEvent(clickEvent);
+  }
+
+  function getCurrentSheet(gadget) {
+    var worksheet = gadget.element.querySelector('.selected')
+          .getAttribute('data-spreadsheet');
+    return gadget.element.querySelector('.spreadsheet').jexcel[worksheet];
+  }
+
+  function buildSelectOptions() {
+    var frag = document.createDocumentFragment(),
+      formulas;
+    formulas = ["SUM", "MIN", "MAX", "COUNT", "AVERAGE", "FLOOR", "ABS",
+                "SQRT", "ISEVEN", "ISODD", "TODAY", "UPPER", "LOWER", "TRUNC",
+                "TYPE", "TRIM", "SIN", "COS", "TAN", "ARCSIN", "ARCCOS",
+                "ARCTAN", "ROUND", "RAND", "RANDBETWEEN", "RADIANS", "POWER",
+                "PI", "PHI", "MOD", "LEN", "LN", "LOG", "LOG10", "FACT", "TRUE",
+                "FALSE", "AND", "OR", "XOR", "EVEN", "ODD", "EXP",
+                "CONCATENATE", "BITAND", "BITOR", "BIN2DEC", "BIN2HEX",
+                "BIN2OCT", "DEC2BIN", "DEC2HEX", "DEC2OCT", "HEX2BIN",
+                "HEX2DEC", "HEX2OCT", "NOT", "OCT2BIN", "OCT2DEC", "OCT2HEX",
+                "PRODUCT", "QUOTIENT", "COLUMN", "ROW", "CELL"].sort();
+    frag.appendChild(
+      domsugar("option", {text: "FORMULA", "class": "formula_option"})
+    );
+    formulas.forEach(function (value) {
+      frag.appendChild(
+        domsugar("option",
+                 {"class": "formula_option", text: value + "()", value: value}
+                )
+      );
+    });
+    return frag;
+  }
+
+  function setupTable(gadget, element) {
+    var filter = element.querySelector(".jexcel_filter"),
+      formula_div = domsugar("div", {"class": "jexcel_formula"}),
+      img = domsugar("img", {src: "fx.png"}),
+      formula_input = domsugar("input", {"class": "jexcel_formula"}),
+      cell_input = domsugar("input", {"class": "cell_input"}),
+      options = buildSelectOptions(),
+      select = domsugar("select", {"class": "minimize"}),
+      buttons;
+    element.querySelector("table.jexcel tr").childNodes.forEach(function (td) {
+      td.style.textAlign = "center";
+    });
+    element.querySelector(".jexcel_toolbar").appendChild(filter);
+    element.querySelector("select.jexcel_toolbar_item")
+      .classList.add("minimize");
+    formula_div.appendChild(img);
+    formula_div.appendChild(formula_input);
+    element.querySelector("div.jexcel_toolbar").parentNode
+      .insertBefore(formula_div,
+                    element.querySelector("div.jexcel_toolbar").nextSibling);
+    formula_input.onfocus = gadget.triggerOnFocusFormulaInput.bind(gadget);
+    formula_input.oninput = function () {
+      return gadget.triggerOnInputFormulaInput(cell_input, this);
+    };
+    cell_input.onfocus = gadget.triggerOnFocusCellInput.bind(gadget);
+    cell_input.onkeypress = function (event) {
+      return gadget.triggerOnKeyPressCellInput(event, cell_input);
+    };
+    formula_div.insertBefore(cell_input, img);
+    select.appendChild(options);
+    select.onchange = function () {
+      var dropdown = this;
+      return gadget.triggerOnChangeSelect(dropdown, formula_input);
+    };
+    element.querySelector(".jexcel_toolbar").insertBefore(select, filter);
+    gadget.element.querySelectorAll(".jexcel_tab_link")
+      .forEach(function (tab, i) {
+        if (i === 0) {
+          gadget.state.selectedTabLink = tab;
+        }
+        tab.title = "Click to rename when selected";
+      });
+    buttons = domsugar("div", {"class": "add_delete"});
+    buttons.appendChild(element.querySelector("i[title='Add table']"));
+    buttons.appendChild(element.querySelector("i[title='Delete table']"));
+    gadget.element.querySelector(".spreadsheet.jexcel_tabs")
+      .appendChild(buttons);
+    gadget.state.newSheet = false;
+  }
+
+  function bindEvents(gadget, sheet) {
+    sheet.onevent = gadget.triggerOnEventSheet.bind(gadget);
+    sheet.onselection = gadget.triggerOnSelectionSheet.bind(gadget);
+    sheet.oneditionend = gadget.triggerOnEditionEndSheet.bind(gadget);
+    return sheet;
+  }
+
+  function getConfigListFromTables(gadget, list) {
+    var configs = [],
+      dict,
+      tmp;
+    list.forEach(function (table, i) {
+      dict = {};
+      dict = Object.assign(jexcel.createFromTable(table),
+                           gadget.state.template);
+      if (table.classList.contains("jexcel") &&
+          !table.classList.contains("jSheet")) {
+        if (Array.from(table.querySelectorAll("td")).filter(function (td) {
+            return td.hasAttribute("cache");
+          }).length === 0) {
+          tmp = JSON.parse(table.dataset.config);
+          dict.columns = tmp.columns;
+          dict.data = tmp.data;
+        }
+      }
+      dict.sheetName = table.title || "Sheet " + (i + 1);
+      configs.push(dict);
+    });
+    return configs;
+  }
+
+  function getTemplate(gadget) {
+    var toolbar_dict = gadget.state.toolbar_dict,
+      list = [],
+      dict,
+      res;
+    dict = {
+      options: gadget.state.options,
+      undo: {
+        type: 'i',
+        content: 'undo',
+        onclick: gadget.triggerUndo.bind(gadget),
+        tooltip: "Undo"
+      },
+      redo: {
+        type: 'i',
+        content: 'redo',
+        onclick: gadget.triggerRedo.bind(gadget),
+        tooltip: "Redo"
+      },
+      merge: {
+        type: 'i',
+        content: 'table_chart',
+        onclick: gadget.triggerMerge.bind(gadget),
+        tooltip: "Merge"
+      },
+      unmerge: {
+        type: 'i',
+        content: 'close',
+        onclick: gadget.triggerUnmerge.bind(gadget),
+        tooltip: "Unmerge"
+      },
+      destroy_merge: {
+        type: 'i',
+        content: 'cancel',
+        onclick: gadget.triggerDestroyMerge.bind(gadget),
+        tooltip: "Unmerge all"
+      },
+      font_style: {
+        type: 'select',
+        k: 'font-family',
+        v: ['Arial', 'Comic Sans MS', 'Verdana', 'Calibri', 'Tahoma',
+            'Helvetica', 'DejaVu Sans', 'Times New Roman', 'Georgia',
+            'Antiqua']
+      },
+      font_size: {
+        type: 'select',
+        k: 'font-size',
+        v: ['8px', '10px', '12px', '14px', '16px', '18px', '20px',
+            '22px', '24px', '26px', '28px', '30px', '34px', '38px',
+            '42px', '46px', '50px']
+      },
+      text_align_left: {
+        type: 'i',
+        content: 'format_align_left',
+        k: 'text-align',
+        v: 'left',
+        tooltip: "Align left"
+      },
+      text_align_center: {
+        type: 'i',
+        content: 'format_align_center',
+        k: 'text-align',
+        v: 'center',
+        tooltip: "Align center"
+      },
+      text_align_right: {
+        type: 'i',
+        content: 'format_align_right',
+        k: 'text-align',
+        v: 'right',
+        tooltip: "Align right"
+      },
+      text_align_justify: {
+        type: 'i',
+        content: 'format_align_justify',
+        k: 'text-align',
+        v: 'justify',
+        tooltip: "Align justify"
+      },
+      vertical_align_top: {
+        type: 'i',
+        content: 'vertical_align_top',
+        k: 'vertical-align',
+        v: 'top',
+        tooltip: "Align top"
+      },
+      vertical_align_middle: {
+        type: 'i',
+        content: 'vertical_align_center',
+        k: 'vertical-align',
+        v: 'middle',
+        tooltip: "Align center"
+      },
+      vertical_align_bottom: {
+        type: 'i',
+        content: 'vertical_align_bottom',
+        k: 'vertical-align',
+        v: 'bottom',
+        tooltip: "Align bottom"
+      },
+      style_bold: {
+        type: 'i',
+        content: 'format_bold',
+        k: 'font-weight',
+        v: 'bold',
+        tooltip: "Bold"
+      },
+      style_underlined: {
+        type: 'i',
+        content: 'format_underlined',
+        k: 'text-decoration',
+        v: 'underline',
+        tooltip: "Underline"
+      },
+      style_italic: {
+        type: 'i',
+        content: 'format_italic',
+        k: 'font-style',
+        v: 'italic',
+        tooltip: "Italic"
+      },
+      text_color: {
+        type: 'color',
+        content: 'format_color_text',
+        k: 'color',
+        tooltip: "Text color"
+      },
+      background_color: {
+        type: 'color',
+        content: 'format_color_fill',
+        k: 'background-color',
+        tooltip: "Background color"
+      },
+      add: {
+        type: "i",
+        content: "add",
+        onclick: gadget.triggerAddSheet.bind(gadget),
+        tooltip: "Add table"
+      },
+      remove: {
+        type: "i",
+        content: "delete",
+        onclick: gadget.triggerDeleteSheet.bind(gadget),
+        tooltip: "Delete table"
+      },
+      add_row: {
+        type: "i",
+        content: "playlist_add",
+        onclick: gadget.triggerAddRow.bind(gadget),
+        tooltip: "Add row at the end"
+      },
+      delete_row: {
+        type: "i",
+        content: "delete_sweep",
+        onclick: gadget.triggerDeleteRow.bind(gadget),
+        tooltip: "Delete last row"
+      },
+      add_column: {
+        type: "i",
+        content: "exposure_plus_1",
+        onclick: gadget.triggerAddColumn.bind(gadget),
+        tooltip: "Add column at the end"
+      },
+      delete_column: {
+        type: "i",
+        content: "exposure_neg_1",
+        onclick: gadget.triggerDeleteColumn.bind(gadget),
+        tooltip: "Delete last column"
+      },
+      dimensions: {
+        type: "i",
+        content: "photo_size_select_small",
+        onclick: gadget.triggerNewDimensions.bind(gadget),
+        tooltip: "Resize table"
+      },
+      contextMenu: function (obj, x, y) {
+        var items = [];
+        gadget.state.obj = obj;
+        gadget.state.x = x;
+        gadget.state.y = y;
+        if (y === null) {
+           // Insert a new column
+          items.push({
+            title: obj.options.text.insertANewColumnBefore,
+            onclick: gadget.triggerInsertNewColumnBefore.bind(gadget)
+          });
+          items.push({
+            title: obj.options.text.insertANewColumnAfter,
+            onclick: gadget.triggerInsertNewColumnAfter.bind(gadget)
+          });
+          // Delete a column
+          items.push({
+            title: obj.options.text.deleteSelectedColumns,
+            onclick: gadget.triggerDeleteSelectedColumns.bind(gadget)
+          });
+          // Rename column
+          items.push({
+            title: obj.options.text.renameThisColumn,
+            onclick: gadget.triggerRenameColumn.bind(gadget)
+          });
+          // Sorting
+          items.push({ type: 'line' });
+          items.push({
+            title: obj.options.text.orderAscending,
+            onclick: gadget.triggerOrderAscending.bind(gadget)
+          });
+          items.push({
+            title: obj.options.text.orderDescending,
+            onclick: gadget.triggerOrderDescending.bind(gadget)
+          });
+        } else {
+          // Insert new row
+          items.push({
+            title: obj.options.text.insertANewRowBefore,
+            onclick: gadget.triggerInsertNewRowBefore.bind(gadget)
+          });
+          items.push({
+            title: obj.options.text.insertANewRowAfter,
+            onclick: gadget.triggerInsertNewRowAfter.bind(gadget)
+          });
+          items.push({
+            title: obj.options.text.deleteSelectedRows,
+            onclick: gadget.triggerDeleteSelectedRows.bind(gadget)
+          });
+        }
+        if (x) {
+          items.push({type: 'line'});
+          items.push({
+            title: "Set column type: Text",
+            onclick: function () {
+              return gadget.triggerChangeType("text");
+            }
+          });
+          items.push({
+            title: "Set column type: Image",
+            onclick: function () {
+              return gadget.triggerChangeType("image");
+            }
+          });
+          items.push({
+            title: "Set column type: HTML",
+            onclick: function () {
+              return gadget.triggerChangeType("html");
+            }
+          });
+          items.push({
+            title: "Set column type: Checkbox",
+            onclick: function () {
+              var child = domsugar("input", {type: "checkbox", name: "c" + x});
+              return gadget.triggerChangeType("checkbox", child);
+            }
+          });
+          items.push({
+            title: "Set column type: Calendar",
+            onclick: function () {
+              return gadget.triggerChangeType("calendar");
+            }
+          });
+          items.push({
+            title: "Set column type: Color",
+            onclick: function () {
+              return gadget.triggerChangeType("color", null, "square");
+            }
+          });
+        }
+        return items;
+      }
+    };
+    if (toolbar_dict.hasOwnProperty("undo_redo") && toolbar_dict.undo_redo) {
+      list.push(dict.undo, dict.redo);
+    }
+    if (toolbar_dict.hasOwnProperty("add") && toolbar_dict.add) {
+      list.push(dict.add, dict.remove);
+    }
+    if (toolbar_dict.hasOwnProperty("merge") && toolbar_dict.merge) {
+      list.push(dict.merge, dict.unmerge, dict.destroy_merge);
+    }
+    if (toolbar_dict.hasOwnProperty("text_font") && toolbar_dict.text_font) {
+      list.push(dict.font_style, dict.font_size, dict.style_bold,
+              dict.style_italic, dict.style_underlined);
+    }
+    if (toolbar_dict.hasOwnProperty("text_position") &&
+        toolbar_dict.text_position) {
+      list.push(dict.text_align_left, dict.text_align_center,
+              dict.text_align_right, dict.text_align_justify,
+              dict.vertical_align_top, dict.vertical_align_middle,
+              dict.vertical_align_bottom);
+    }
+    if (toolbar_dict.hasOwnProperty("color_picker") &&
+        toolbar_dict.color_picker) {
+      list.push(dict.text_color, dict.background_color);
+    }
+    if (isMobileDevice()) {
+      list.push({
+        type: "i",
+        content: "photo_library",
+        onclick: function (sheet, instance) {
+          return gadget.triggerChangeTypeInToolbar(sheet, instance, "image");
+        }
+      });
+      list.push({
+        type: "i",
+        content: "format_size",
+        onclick: function (sheet, instance) {
+          return gadget.triggerChangeTypeInToolbar(sheet, instance, "text");
+        }
+      });
+      list.push({
+        type: "i",
+        content: "format_paint",
+        onclick: function (sheet, instance) {
+          return gadget.triggerChangeTypeInToolbar(sheet, instance,
+                                                   "color", null, "square");
+        }
+      });
+      list.push({
+        type: "i",
+        content: "format_list_bulleted",
+        onclick: function (sheet, instance) {
+          return gadget.triggerChangeTypeInToolbar(sheet, instance, "html");
+        }
+      });
+      list.push({
+        type: "i",
+        content: "calendar_today",
+        onclick: function (sheet, instance) {
+          return gadget.triggerChangeTypeInToolbar(sheet, instance, "calendar");
+        }
+      });
+      list.push({
+        type: "i",
+        content: "check_box",
+        onclick: function (sheet, instance) {
+          var child = domsugar("input", {type: "checkbox"});
+          return gadget.triggerChangeTypeInToolbar(sheet, instance,
+                                                   "checkbox", child);
+        }
+      });
+    }
+    if (toolbar_dict.hasOwnProperty("add_delete_row_column") &&
+        toolbar_dict.add_delete_row_column) {
+      list.push(dict.add_row, dict.delete_row,
+              dict.add_column, dict.delete_column, dict.dimensions);
+    }
+    res = Object.assign({}, dict.options);
+    res.toolbar = list;
+    res.contextMenu = dict.contextMenu;
+    return res;
+  }
+
+  function setHistoryType(instance, action, column, old_type, new_type) {
+    instance.setHistory({
+      action: action,
+      column: column,
+      oldType: old_type,
+      newType: new_type
+    });
+  }
+
+  rJS(window)
+
+    .setState({
+      saveConfig: false,
+      newSheet: false,
+      updateSelection: true,
+      toolbar_dict: {
+        undo_redo: true,
+        add: true,
+        merge: true,
+        text_font: true,
+        text_position: true,
+        color_picker: true,
+        add_delete_row_column: true
+      },
+      options: {
+        minDimensions: [26, 100],
+        defaultColWidth: 100,
+        defaultColAlign: "left",
+        allowExport: true,
+        columnSorting: true,
+        columnDrag: true,
+        columnResize: true,
+        rowResize: true,
+        rowDrag: true,
+        editable: true,
+        allowInsertRow: true,
+        allowManualInsertRow: true,
+        allowInsertColumn: true,
+        allowManualInsertColumn: true,
+        allowDeleteRow: true,
+        allowRenameColumn: true,
+        selectionCopy: true,
+        search: true,
+        fullscreen: true,
+        autoIncrement: true,
+        parseFormulas: true,
+        wordWrap: true
+      }
+    })
+
+    .declareAcquiredMethod("notifyChange", "notifyChange")
+
+    .declareMethod("render", function (options) {
+      return this.changeState(options);
+    })
+
+    .declareMethod('getContent', function () {
+      var gadget = this,
+        form_data = {},
+        res = "",
+        sheets = gadget.element.querySelector('.spreadsheet').jexcel,
+        tab_links = gadget.element.querySelectorAll('.jexcel_tab_link');
+      if (this.state.editable) {
+        sheets.forEach(function (sheet, i) {
+          var table = sheet.el.querySelector("table.jexcel").cloneNode(true),
+            config = sheet.getConfig(),
+            dict;
+          dict = {
+            columns: config.columns,
+            data: config.data
+          };
+          table.dataset.config = JSON.stringify(dict);
+          table.title = tab_links[i].textContent;
+          table.border = "1px";
+          table.querySelector("colgroup col").remove();
+          table.querySelector("tr").remove();
+          table.querySelectorAll("td.jexcel_row").forEach(function (td) {
+            td.remove();
+          });
+          table.querySelectorAll('tr').forEach(function (tr, y) {
+            tr.childNodes.forEach(function (td, x) {
+              if (td.textContent !== "") {
+                var value = sheet.getValueFromCoords(x, y);
+                if (value[0] === "=") {
+                  td.dataset.formula = value;
+                  td.setAttribute('formula', value);
+                }
+              }
+            });
+          });
+          res += table.outerHTML;
+        });
+        form_data[this.state.key] = beautifyHTMLString(res);
+      }
+      return form_data;
+    })
+
+    .onStateChange(function (modification_dict) {
+      var gadget = this, tabs, i,
+        toolbar_config,
+        toolbar_events_config,
+        nodes,
+        data_list,
+        selected;
+      if (modification_dict.hasOwnProperty('newSheet') &&
+          modification_dict.newSheet) {
+        tabs = (gadget.element.querySelectorAll(".jexcel_container"));
+        return setupTable(gadget, tabs[tabs.length - 1]);
+      }
+      if (modification_dict.hasOwnProperty('value')) {
+        toolbar_config = getTemplate(gadget);
+        toolbar_events_config = bindEvents(gadget, toolbar_config);
+        if (gadget.state.value === "") {
+          toolbar_events_config.sheetName = "Table 1";
+          jexcel.tabs(gadget.element.querySelector(".spreadsheet"),
+                      [toolbar_events_config]
+                     );
+          gadget.state.tables =
+             [gadget.element.querySelector("div.jexcel_content > table")];
+          gadget.element
+            .querySelectorAll(".jexcel_container")
+            .forEach(function (tab) {
+              return setupTable(gadget, tab);
+            });
+        } else {
+          gadget.state.tables = [];
+          nodes = createElementFromHTML(gadget.state.value);
+          for (i = 0; i < nodes.length; i += 1) {
+            gadget.state.tables[i] = nodes[i];
+          }
+          data_list = getConfigListFromTables(gadget, gadget.state.tables);
+          data_list.forEach(function (dict) {
+            Object.assign(dict, toolbar_events_config);
+          });
+          jexcel.tabs(gadget.element.querySelector(".spreadsheet"), data_list);
+          gadget.element.querySelectorAll(".jexcel_container")
+            .forEach(function (tab) {
+              return setupTable(gadget, tab);
+            });
+          selected = gadget.element
+            .querySelector('.jexcel_tab_link.selected');
+          selected.classList.remove("selected");
+          gadget.element.querySelector('.jexcel_tab_link')
+            .classList.add("selected");
+          gadget.element.querySelectorAll(".jexcel_container")
+            .forEach(function (tab, i) {
+              if (i === 0) {
+                tab.style.display = "block";
+              } else {
+                tab.style.display = "none";
+              }
+            });
+        }
+      }
+    })
+
+    .declareJob("deferNotifyChange", function () {
+      return this.notifyChange();
+    })
+
+    .declareJob("triggerAddRow", function (sheet, instance) {
+      instance.insertRow();
+    })
+
+    .declareJob("triggerDeleteRow", function (sheet, instance) {
+      instance.deleteRow(instance.options.data.length - 1, 1);
+    })
+
+    .declareJob("triggerInsertNewColumnBefore", function () {
+      var state = this.state;
+      state.obj.insertColumn(1, parseInt(state.x, 10), 1);
+    })
+
+    .declareJob("triggerInsertNewColumnAfter", function () {
+      var state = this.state;
+      state.obj.insertColumn(1, parseInt(state.x, 10), 0);
+    })
+
+    .declareJob("triggerDeleteSelectedColumns", function () {
+      var state = this.state;
+      state.obj.deleteColumn(state.obj.getSelectedColumns().length ?
+                           undefined : parseInt(state.x, 10));
+    })
+
+    .declareJob("triggerRenameColumn", function () {
+      var state = this.state;
+      state.obj.setHeader(state.x);
+    })
+
+    .declareJob("triggerOrderAscending", function () {
+      var state = this.state;
+      state.obj.orderBy(state.x, 0);
+    })
+
+    .declareJob("triggerOrderDescending", function () {
+      var state = this.state;
+      state.obj.orderBy(state.x, 1);
+    })
+
+    .declareJob("triggerInsertNewRowBefore", function () {
+      var state = this.state;
+      state.obj.insertRow(1, parseInt(state.y, 10), 1);
+    })
+
+    .declareJob("triggerInsertNewRowAfter", function () {
+      var state = this.state;
+      state.obj.insertRow(1, parseInt(state.y, 10));
+    })
+
+    .declareJob("triggerDeleteSelectedRows", function () {
+      var state = this.state;
+      state.obj.deleteRow(state.obj.getSelectedRows().length ?
+                        undefined : parseInt(state.y, 10));
+    })
+
+    .declareJob("triggerAddColumn", function (sheet, instance) {
+      instance.insertColumn();
+    })
+
+    .declareJob("triggerDeleteColumn", function (sheet, instance) {
+      instance.deleteColumn(instance.options.columns.length - 1, 1);
+    })
+
+    .declareJob("triggerUndo", function (sheet, instance) {
+      instance.undo();
+    })
+
+    .declareJob("triggerRedo", function (sheet, instance) {
+      instance.redo();
+    })
+
+    .declareJob("triggerMerge", function (sheet, instance) {
+      var cell = sheet.querySelector("td.highlight"),
+        selected = instance.getJson(true),
+        colspan = Object.keys(selected[0]).length,
+        rowspan = selected.length,
+        coor = getCoordinatesFromCell(cell);
+      instance.setMerge(coor, colspan, rowspan);
+    })
+
+    .declareJob("triggerUnmerge", function (sheet, instance) {
+      var cell = document.querySelector("td.highlight-selected"),
+        coor = getCoordinatesFromCell(cell);
+      instance.removeMerge(coor);
+    })
+
+    .declareJob("triggerDestroyMerge", function (sheet, instance) {
+      instance.destroyMerged();
+    })
+
+    .declareJob("triggerChangeType", function (type, child, render) {
+      var state = this.state,
+        x = parseInt(state.x, 10),
+        y = state.y !== null ? parseInt(state.y, 10) : 0,
+        cell,
+        column,
+        array;
+      state.obj.updateSelectionFromCoords(x, y, x, y);
+      cell = state.obj.el.querySelector("td.highlight-selected");
+      if (state.obj.options.columns[x].type !== type) {
+        column = state.obj.el
+          .querySelectorAll("td[data-x='" + x + "']");
+        array = Array.from(column);
+        array.shift();
+        setHistoryType(state.obj, "beginChangeType",
+                       x,
+                       state.obj.options.columns[x].type,
+                       type);
+        array.forEach(function (cell) {
+          state.obj.setValue(getCoordinatesFromCell(cell), "");
+          cell.innerHTML = "";
+          if (child) {
+            cell.appendChild(child.cloneNode());
+          }
+        });
+        setHistoryType(state.obj, "endChangeType",
+                       x,
+                       state.obj.options.columns[x].type,
+                       type);
+        state.obj.options.columns[x].type = type;
+        if (render) {
+          state.obj.options.columns[x].render = render;
+        }
+        fireDoubleClick(cell);
+      }
+    })
+
+    .declareJob("triggerChangeTypeInToolbar",
+                function (sheet, instance, type, child, render) {
+        var cell = sheet.querySelector("td.highlight-selected"),
+          x,
+          column,
+          array;
+        x = cell ? parseInt(cell.dataset.x, 10) : null;
+        if (cell && instance.options.columns[x].type !== type) {
+          column = sheet.querySelectorAll("td[data-x='" + x + "']");
+          array = Array.from(column);
+          array.shift();
+          setHistoryType(instance, "beginChangeType",
+                         x,
+                         instance.options.columns[x].type,
+                         type);
+          array.forEach(function (cell) {
+            instance.setValue(getCoordinatesFromCell(cell), "");
+            cell.innerHTML = "";
+            if (child) {
+              cell.appendChild(child.cloneNode());
+            }
+          });
+          setHistoryType(instance, "endChangeType",
+                         x,
+                         instance.options.columns[x].type,
+                         type);
+          instance.options.columns[x].type = type;
+          if (render) {
+            instance.options.columns[x].render = render;
+          }
+          fireDoubleClick(cell);
+        }
+      })
+
+    .declareJob("triggerNewDimensions", function (sheet, instance) {
+      var r = prompt("Number of rows :", instance.options.data.length),
+        c = prompt("Number of columns :", instance.options.columns.length);
+      if (c > 0 && r > 0) {
+        instance.setHistory({action: "beginResizeTable"});
+        if (c > instance.options.columns.length) {
+          while (instance.options.columns.length < c) {
+            instance.insertColumn();
+          }
+        } else {
+          while (instance.options.columns.length > c) {
+            instance.deleteColumn(instance.options.columns.length - 1, 1);
+          }
+        }
+        if (r > instance.options.data.length) {
+          while (instance.options.data.length < r) {
+            instance.insertRow();
+          }
+        } else {
+          while (instance.options.data.length > r) {
+            instance.deleteRow(instance.options.data.length - 1, 1);
+          }
+        }
+        instance.setHistory({action: "endResizeTable"});
+      }
+    })
+
+    .declareJob("triggerAddSheet", function () {
+      var gadget = this,
+        tabs = gadget.element.querySelectorAll(".jexcel_tab_link"),
+        dict1,
+        dict2;
+      if (tabs.length === 16) {
+        alert("Can't add tables anymore.");
+      } else {
+        dict1 = getTemplate(gadget);
+        dict1.sheetName = "Table " +
+          (gadget.element.querySelector('.spreadsheet').jexcel.length + 1);
+        dict2 = bindEvents(gadget, dict1);
+        jexcel.tabs(gadget.element.querySelector(".spreadsheet"), [dict2]);
+        gadget.element
+          .querySelectorAll('.jexcel_container')[gadget.element.querySelector('.spreadsheet').jexcel.length - 1]
+          .querySelectorAll("td[data-x][data-y]")
+          .forEach(function (td) {
+            td.style.textAlign = "left";
+          });
+        gadget.deferNotifyChange();
+        return gadget.changeState({newSheet: true})
+          .push(function () {
+            gadget.state.selectedTabLink = gadget.element
+              .querySelector(".jexcel_tab_link.selected");
+          });
+      }
+    })
+
+    .declareJob("triggerDeleteSheet", function (sheet, instance) {
+      var gadget = this,
+        tab_link = gadget.element
+          .querySelector('.jexcel_tab_link.selected'),
+        index = tab_link.getAttribute("data-spreadsheet"),
+        to_remove,
+        sheets;
+      if (confirm("Delete this table ?")) {
+        if (gadget.element.querySelector('.spreadsheet').jexcel.length > 1) {
+          tab_link.remove();
+          gadget.element.querySelectorAll(".jexcel_container")
+            .forEach(function (tab) {
+              if (tab.style.display === "block") {
+                to_remove = tab;
+              }
+            });
+          to_remove.remove();
+          gadget.element.querySelector('.spreadsheet').jexcel.splice(index, 1);
+          sheets = gadget.element.querySelectorAll('.jexcel_container');
+          sheets[sheets.length - 1].style.display = "block";
+          gadget.element.querySelectorAll('.jexcel_tab_link')
+              .forEach(function (tab, i) {
+              if (i === sheets.length - 1) {
+                tab.classList.add("selected");
+              }
+              tab.dataset.spreadsheet = i;
+              tab.textContent = tab.textContent.substring(0, 5) === "Table" ?
+                  "Table " + (i + 1) : tab.textContent;
+            });
+        } else {
+          gadget.element.querySelector('.jexcel_tab_link')
+            .textContent = "Table 1";
+          sheet.querySelector("input.jexcel_formula").value = "";
+          instance.setData(new Array(instance.options.columns.length)
+            .fill(0, instance.options.columns.length - 1,
+                  new Array(instance.options.data.length)
+                    .fill(0, instance.options.data.length - 1, "")));
+          instance.options.columns.forEach(function (column) {
+            column.type = "text";
+          });
+        }
+        gadget.deferNotifyChange();
+        gadget.state.selectedTabLink = gadget.element
+          .querySelector(".jexcel_tab_link.selected");
+      }
+    })
+
+    .declareJob("triggerOnFocusFormulaInput", function () {
+      var gadget = this,
+        worksheet = gadget.element.querySelector('.selected')
+          .getAttribute('data-spreadsheet');
+      gadget.element.querySelector('.spreadsheet').jexcel[worksheet]
+        .resetSelection(true);
+    })
+
+    .declareJob("triggerOnInputFormulaInput",
+              function (cell_input, formula_input) {
+        var gadget = this,
+          worksheet = gadget.element.querySelector('.selected')
+            .getAttribute('data-spreadsheet'),
+          instance = gadget.element.querySelector('.spreadsheet')
+            .jexcel[worksheet],
+          e = formula_input.value,
+          numbers = ["0", "1", "2", "3", "4", "5", "6", "7", "8", "9"];
+        if (e[0] === "=" && e[e.length - 1] !== ")") {
+          if (numbers.includes(e[e.length - 1])) {
+            instance.setValue(cell_input.value, e);
+          }
+        } else {
+          instance.setValue(cell_input.value, e);
+        }
+      })
+
+    .declareJob("triggerOnFocusCellInput", function () {
+      var gadget = this,
+        worksheet = gadget.element.querySelector('.selected')
+          .getAttribute('data-spreadsheet');
+      gadget.element.querySelector('.spreadsheet').jexcel[worksheet]
+          .resetSelection(true);
+    })
+
+    .declareJob("triggerOnKeyPressCellInput", function (event, input) {
+      var gadget = this,
+        worksheet,
+        x,
+        y,
+        ys;
+      if (event.keyCode === 13) {
+        worksheet = gadget.element.querySelector('.selected')
+            .getAttribute('data-spreadsheet');
+        y = input.value.match(/(\d+)/)[0];
+        x = columnLetterToNumber(input.value
+                               .substring(0, input.value.length - y.length)
+                              );
+        ys = parseInt(y, 10) - 1;
+        gadget.element.querySelector('.spreadsheet').jexcel[worksheet]
+          .updateSelectionFromCoords(x, ys, x, ys);
+      }
+    })
+
+    .declareJob("triggerOnChangeSelect", function (dropdown, formula_input) {
+      var gadget = this,
+        sheet = getCurrentSheet(gadget),
+        cell = sheet.el.querySelector("td.highlight-selected"),
+        x,
+        y,
+        value,
+        currentValue;
+      if (cell && sheet.options.columns[Number(cell.dataset.x)]
+          .type === "text") {
+        x = Number(cell.dataset.x);
+        y = Number(cell.dataset.y);
+        currentValue = sheet.getValueFromCoords(x, y);
+        if (currentValue === "" || currentValue[0] !== "=") {
+          value = "=" +
+            dropdown.options[dropdown.selectedIndex].value +
+            "(" + currentValue + ")";
+        } else {
+          value = "=" +
+            dropdown.options[dropdown.selectedIndex].value +
+            "(" + currentValue.substring(1, currentValue.length) + ")";
+        }
+        sheet.setValueFromCoords(x, y, value);
+        formula_input.value = value;
+      }
+      dropdown.selectedIndex = 0;
+    })
+
+    .declareJob("triggerOnEventSheet", function (event) {
+      var gadget = this,
+        exluded_events = ["onload", "onfocus", "onblur", "onselection"];
+      if (!exluded_events.includes(event)) {
+        if ((["onchangestyle", "onchange", "onbeforechange"].includes(event) &&
+             gadget.state.saveConfig) ||
+             !["onchangestyle", "onchange", "onbeforechange"].includes(event)) {
+          gadget.deferNotifyChange();
+        }
+      }
+    })
+
+    .declareJob("triggerOnSelectionSheet", function () {
+      var gadget = this,
+        instance = getCurrentSheet(gadget),
+        tab = gadget.element
+             .querySelectorAll(".jexcel_container")[gadget.element
+                                                 .querySelector("div.jexcel_tab_link.selected")
+                                                 .getAttribute("data-spreadsheet")],
+        cell = tab.querySelector("td.highlight-selected"),
+        cell_input = tab.querySelector("input.cell_input"),
+        formula = tab.querySelector("input.jexcel_formula"),
+        x,
+        y;
+      cell_input.value = getCoordinatesFromCell(cell);
+      x = Number(cell.dataset.x);
+      y = Number(cell.dataset.y);
+      formula.value = ["text", "calendar", "checkbox", "color"]
+        .includes(instance.options.columns[x].type) ?
+            instance.getValueFromCoords(x, y) : "";
+      if (instance.options.columns[x].type === "text") {
+        formula.readOnly = false;
+        formula.classList.remove("readonly");
+      } else {
+        formula.readOnly = true;
+        formula.classList.add("readonly");
+      }
+    })
+
+    .declareJob("triggerOnEditionEndSheet", function (cell, x, y, value) {
+      var gadget = this,
+        numbers,
+        worksheet,
+        tab;
+      if (value) {
+        if (value[0] === "=" && value[value.length - 1] !== ")") {
+          numbers = ["0", "1", "2", "3", "4", "5", "6", "7", "8", "9"];
+          if (numbers.includes(value[value.length - 1])) {
+            worksheet = gadget.element.querySelector('.selected')
+              .getAttribute('data-spreadsheet');
+            tab = gadget.element.querySelector('.spreadsheet')
+              .jexcel[worksheet];
+            tab.setValueFromCoords(x, y, value);
+          } else {
+            cell.textContent = value;
+          }
+        }
+      }
+    })
+
+    .onEvent("input", function (ev) {
+      var gadget = this, sheet, formula, td;
+      sheet = getCurrentSheet(gadget);
+      td = sheet.el.querySelector("td.highlight-selected");
+      if (td && ev.target === td.childNodes[0]) {
+        formula = sheet.el.querySelector("input.jexcel_formula");
+        formula.value = ev.target.value;
+      }
+    }, false, false)
+
+    .onEvent("click", function (ev) {
+      var gadget = this, name;
+      gadget.state.saveConfig = true;
+      if (ev.target.classList.contains("jexcel_tab_link")) {
+        if (ev.target === gadget.state.selectedTabLink) {
+          name = prompt("Table name :", ev.target.textContent);
+          gadget.state.selectedTabLink.textContent = name !== null ?
+              name : gadget.state.selectedTabLink.textContent;
+          gadget.deferNotifyChange();
+        }
+        gadget.state.selectedTabLink = ev.target;
+      }
+    }, false, false);
+
+}(window, rJS, jexcel, domsugar, document, alert, prompt, confirm, navigator));
\ No newline at end of file
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.js.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.js.xml
new file mode 100644
index 0000000000000000000000000000000000000000..5c8cbcd107f645a56268e56cbeaabeb3384420ef
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.gadget.js.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>jexcel.gadget.js</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>application/javascript</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.xml
new file mode 100644
index 0000000000000000000000000000000000000000..4075101f339a97ef2cb163fdc1ca551d6c407a9f
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel.xml
@@ -0,0 +1,26 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="Folder" module="OFS.Folder"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>_objects</string> </key>
+            <value>
+              <tuple/>
+            </value>
+        </item>
+        <item>
+            <key> <string>id</string> </key>
+            <value> <string>jexcel</string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.css.css b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.css.css
new file mode 100644
index 0000000000000000000000000000000000000000..e848aa2b56053bf636ba40b3b1ed9342168f1bd9
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.css.css
@@ -0,0 +1,740 @@
+:root {
+    --jexcel-border-color:#000;
+}
+
+.jexcel_container {
+    display:inline-block;
+    padding-right:2px;
+    box-sizing: border-box;
+    overscroll-behavior: contain;
+    padding-bottom: 33px;
+}
+
+.jexcel_container.fullscreen {
+    position:fixed;
+    top:38px;
+    left:0px;
+    width:100%;
+    height:100%;
+    z-index:21;
+}
+
+.jexcel_container.fullscreen .jexcel_content {
+    overflow:auto;
+    width:100%;
+    height:96.5%;
+    background-color:#ffffff;
+}
+
+.jexcel_container.fullscreen.with-toolbar {
+    height: calc(100% - 46px);
+}
+
+.jexcel_content {
+    display:inline-block;
+    box-sizing: border-box;
+    padding-right:3px;
+    padding-bottom:3px;
+    position:relative;
+    scrollbar-width: thin;
+    scrollbar-color: #666 transparent;
+}
+
+@supports (-moz-appearance:none) {
+    .jexcel_content { padding-right:10px; } 
+}
+
+.jexcel_content::-webkit-scrollbar {
+    width: 5px;
+    height: 5px;
+}
+
+.jexcel_content::-webkit-scrollbar-track {
+    background: #eee;
+}
+ 
+.jexcel_content::-webkit-scrollbar-thumb {
+  background: #666; 
+}
+
+.jexcel {
+    border-collapse: separate;
+    table-layout: fixed;
+    white-space:  nowrap;
+    empty-cells: show;
+    border: 0px;
+    background-color: #fff;
+    width: 0;
+
+    /*border-top: 1px solid transparent;
+    border-left: 1px solid transparent;
+    border-right: 1px solid #ccc;
+    border-bottom: 1px solid #ccc;*/
+}
+
+.jexcel > thead > tr > td
+{
+    border-top: 1px solid #ccc;
+    border-left: 1px solid #ccc;
+    border-right: 1px solid transparent;
+    border-bottom: 1px solid transparent;
+    background-color: #f3f3f3;
+    padding: 2px;
+    cursor: pointer;
+    box-sizing: border-box;
+    overflow: hidden;
+    position: -webkit-sticky;
+    position: sticky;
+    top: 0;
+    z-index:2;
+}
+
+.with-toolbar .jexcel > thead > tr > td
+{
+    /**top:42px;**/
+}
+
+.jexcel > thead > tr > td.dragging
+{
+    background-color:#fff;
+    opacity:0.5;
+}
+
+.jexcel > thead > tr > td.selected
+{
+    background-color:#dcdcdc;
+}
+
+.jexcel > thead > tr > td.arrow-up
+{
+    background-repeat:no-repeat;
+    background-position:center right 5px;
+    background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3Cpath d='M7 14l5-5 5 5H7z' fill='gray'/%3E%3C/svg%3E");
+    text-decoration:underline;
+}
+
+.jexcel > thead > tr > td.arrow-down
+{
+    background-repeat:no-repeat;
+    background-position:center right 5px;
+    background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3Cpath d='M7 10l5 5 5-5H7z' fill='gray'/%3E%3C/svg%3E");
+    text-decoration:underline;
+}
+
+.jexcel > tbody > tr > td:first-child
+{
+    position:sticky;
+    left:0;
+    background-color:#f3f3f3;
+    text-align:center;
+}
+
+.jexcel > tbody.resizable > tr > td:first-child::before
+{
+    content:'\00a0';
+    width:100%;
+    height:3px;
+    position:absolute;
+    bottom:0px;
+    left:0px;
+    cursor:row-resize;
+}
+
+.jexcel > tbody.draggable > tr > td:first-child::after
+{
+    content:'\00a0';
+    width:3px;
+    height:100%;
+    position:absolute;
+    top:0px;
+    right:0px;
+    cursor:move;
+}
+
+.jexcel > tbody > tr.dragging > td
+{
+    background-color:#eee;
+    opacity:0.5;
+}
+
+.jexcel > tbody > tr > td
+{
+    border-top:1px solid #ccc;
+    border-left:1px solid #ccc;
+    border-right:1px solid transparent;
+    border-bottom:1px solid transparent;
+    padding:4px;
+    white-space: nowrap;
+    box-sizing: border-box;
+    line-height:1em;
+}
+
+.jexcel > tbody > tr > td:last-child
+{
+    overflow:hidden;
+}
+
+.jexcel > tbody > tr > td > img
+{
+    display:inline-block;
+    max-width:100px;
+}
+
+.jexcel > tbody > tr > td.readonly
+{
+    color:rgba(0,0,0,0.3)
+}
+.jexcel > tbody > tr.selected > td:first-child
+{
+    background-color:#dcdcdc;
+}
+.jexcel > tbody > tr > td > select,
+.jexcel > tbody > tr > td > input,
+.jexcel > tbody > tr > td > textarea
+{
+    border:0px;
+    border-radius:0px;
+    outline:0px;
+    width:100%;
+    margin:0px;
+    padding:0px;
+    background-color:transparent;
+    box-sizing: border-box;
+}
+
+.jexcel > tbody > tr > td > textarea
+{
+    resize: none;
+    padding-top:6px !important;
+}
+
+.jexcel > tbody > tr > td > input[type=checkbox]
+{
+    width:12px;
+    margin-top:2px;
+}
+.jexcel > tbody > tr > td > input[type=radio]
+{
+    width:12px;
+    margin-top:2px;
+}
+
+.jexcel > tbody > tr > td > select
+{
+    -webkit-appearance: none;
+    -moz-appearance: none;
+    appearance: none;
+    background-repeat: no-repeat;
+    background-position-x: 100%;
+    background-position-y: 40%;
+    background-image: url();
+}
+
+.jexcel > tbody > tr > td.jexcel_dropdown
+{
+    background-repeat: no-repeat;
+    background-position:top 50% right 5px;
+    background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3Cpath d='M7 10l5 5 5-5H7z' fill='lightgray'/%3E%3C/svg%3E");
+    text-overflow: ellipsis;
+    overflow-x:hidden;
+}
+
+.jexcel > tbody > tr > td.jexcel_dropdown.jexcel_comments
+{
+    background:url("') top right no-repeat;
+}
+
+.jexcel > tbody > tr > td > .color
+{
+    width:90%;
+    height:10px;
+    margin:auto;
+}
+
+.jexcel > tbody > tr > td > a {
+    text-decoration: underline;
+}
+
+.jexcel > tbody > tr > td.highlight > a {
+    color: blue;
+    cursor: pointer;
+}
+
+.jexcel > tfoot > tr > td
+{
+    border-top: 1px solid #ccc;
+    border-left: 1px solid #ccc;
+    border-right: 1px solid transparent;
+    border-bottom: 1px solid transparent;
+    background-color: #f3f3f3;
+    padding: 2px;
+    cursor: pointer;
+    box-sizing: border-box;
+    overflow: hidden;
+}
+
+.jexcel .highlight {
+    background-color:rgba(0,0,0,0.05);
+}
+
+.jexcel .highlight-top {
+    border-top:1px solid #000; /* var(--jexcel-border-color);*/
+    box-shadow: 0px -1px #ccc;
+}
+
+.jexcel .highlight-left {
+    border-left:1px solid #000; /* var(--jexcel-border-color);*/
+    box-shadow: -1px 0px #ccc;
+}
+
+.jexcel .highlight-right {
+    border-right:1px solid #000; /* var(--jexcel-border-color);*/
+}
+
+.jexcel .highlight-bottom {
+    border-bottom:1px solid #000; /* var(--jexcel-border-color);*/
+}
+
+.jexcel .highlight-top.highlight-left {
+    box-shadow: -1px -1px #ccc;
+    -webkit-box-shadow: -1px -1px #ccc;
+    -moz-box-shadow: -1px -1px #ccc;
+}
+
+.jexcel .highlight-selected
+{
+    background-color:rgba(0,0,0,0.0);
+}
+.jexcel .selection
+{
+    background-color:rgba(0,0,0,0.05);
+}
+.jexcel .selection-left
+{
+    border-left:1px dotted #000;
+}
+.jexcel .selection-right
+{
+    border-right:1px dotted #000;
+}
+.jexcel .selection-top
+{
+    border-top:1px dotted #000;
+}
+.jexcel .selection-bottom
+{
+    border-bottom:1px dotted #000;
+}
+.jexcel_corner
+{
+    position:absolute;
+    background-color: rgb(0, 0, 0);
+    height: 1px;
+    width: 1px;
+    border: 1px solid rgb(255, 255, 255);
+    top:-2000px;
+    left:-2000px;
+    cursor:crosshair;
+    box-sizing: initial;
+    z-index:30;
+    padding: 2px;
+}
+
+.jexcel .editor
+{
+    outline:0px solid transparent;
+    overflow:visible;
+    white-space: nowrap;
+    text-align:left;
+    padding:0px;
+    box-sizing: border-box;
+    overflow:visible !important;
+}
+
+.jexcel .editor > input
+{
+    padding-left:4px;
+}
+
+.jexcel .editor .jupload
+{
+    position:fixed;
+    top:100%;
+    z-index:40;
+    user-select:none;
+    -webkit-font-smoothing: antialiased;
+    font-size: .875rem;
+    letter-spacing: .2px;
+    -webkit-border-radius: 4px;
+    border-radius: 4px;
+    -webkit-box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2);
+    box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2);
+    padding:10px;
+    background-color:#fff;
+    width:300px;
+    min-height:225px;
+    margin-top:2px;
+}
+
+.jexcel .editor .jupload img
+{
+    width:100%;
+    height:auto;
+}
+
+.jexcel .editor .jexcel_richtext
+{
+    position:fixed;
+    top:100%;
+    z-index:40;
+    user-select:none;
+    -webkit-font-smoothing: antialiased;
+    font-size: .875rem;
+    letter-spacing: .2px;
+    -webkit-box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2);
+    box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2);
+    padding:10px;
+    background-color:#fff;
+    min-width:280px;
+    max-width:310px;
+    margin-top:2px;
+    text-align:left;
+}
+
+.jexcel .editor .jclose:after
+{
+    position:absolute;
+    top:0;
+    right:0;
+    margin:10px;
+    content:'close';
+    font-family:'Material icons';
+    font-size:24px;
+    width:24px;
+    height:24px;
+    line-height:24px;
+    cursor:pointer;
+    text-shadow: 0px 0px 5px #fff;
+}
+
+.jexcel, .jexcel td, .jexcel_corner
+{
+  -webkit-touch-callout: none;
+  -webkit-user-select: none;
+  -khtml-user-select: none;
+  -moz-user-select: none;
+  -ms-user-select: none;
+  user-select: none;
+  -webkit-user-drag: none;
+  -khtml-user-drag: none;
+  -moz-user-drag: none;
+  -o-user-drag: none;
+  user-drag: none;
+}
+
+.jexcel_textarea
+{
+    position:absolute;
+    top:-999px;
+    left:-999px;
+    width:1px;
+    height:1px;
+}
+.jexcel .dragline
+{
+    position:absolute;
+}
+.jexcel .dragline div
+{
+    position:relative;
+    top:-6px;
+    height:5px;
+    width:22px;
+}
+.jexcel .dragline div:hover
+{
+    cursor:move;
+}
+
+.jexcel .onDrag
+{
+    background-color:rgba(0,0,0,0.6);
+}
+
+.jexcel .error
+{
+    border:1px solid red;
+}
+
+.jexcel thead td.resizing
+{
+    border-right-style:dotted !important;
+    border-right-color:red !important;
+}
+
+.jexcel tbody tr.resizing > td
+{
+    border-bottom-style:dotted !important;
+    border-bottom-color:red !important;
+}
+
+.jexcel tbody td.resizing
+{
+    border-right-style:dotted !important;
+    border-right-color:red !important;
+}
+
+.jexcel .jdropdown-header
+{
+    border:0px !important;
+    outline:none !important;
+    width:100% !important;
+    height:100% !important;
+    padding:0px !important;
+    padding-left:8px !important;
+}
+
+.jexcel .jdropdown-container
+{
+    margin-top:1px;
+}
+
+.jexcel .jdropdown-container-header {
+    padding: 0px;
+    margin: 0px;
+    height: inherit;
+}
+
+.jexcel .jdropdown-picker
+{
+    border:0px !important;
+    padding:0px !important;
+    width:inherit;
+    height:inherit;
+}
+
+.jexcel .jexcel_comments
+{
+    background:url('');
+    background-repeat: no-repeat;
+    background-position: top right;
+}
+
+.jexcel .sp-replacer
+{
+    margin: 2px;
+    border:0px;
+}
+
+.jexcel > thead > tr.jexcel_filter > td > input
+{
+    border:0px;
+    width:100%;
+    outline:none;
+}
+
+.jexcel_about {
+    float: right;
+    font-size: 0.7em;
+    padding: 2px;
+    text-transform: uppercase;
+    letter-spacing: 1px;
+    display: none;
+}
+.jexcel_about a {
+    color: #ccc;
+    text-decoration: none;
+}
+
+.jexcel_about img {
+    display: none;
+}
+
+.jexcel_filter
+{
+    display:flex;
+    justify-content:space-between;
+    /**margin-bottom:4px;**/
+}
+
+.jexcel_filter > div
+{
+    padding:4px;
+    align-items:center;
+}
+
+.jexcel_pagination
+{
+    display:flex;
+    justify-content:space-between;
+    align-items:center;
+}
+
+.jexcel_pagination > div
+{
+    display:flex;
+    padding:10px;
+}
+
+.jexcel_pagination > div:last-child
+{
+    padding-right:10px;
+    padding-top:10px;
+}
+
+.jexcel_pagination > div > div
+{
+    text-align:center;
+    width:36px;
+    height:36px;
+    line-height:34px;
+    border:1px solid #ccc;
+    box-sizing: border-box;
+    margin-left:2px;
+    cursor:pointer;
+}
+
+.jexcel_page
+{
+    font-size:0.8em;
+}
+
+.jexcel_page_selected
+{
+    font-weight:bold;
+    background-color:#f3f3f3;
+}
+
+.jexcel_toolbar
+{
+    display:flex;
+    background-color:#f3f3f3;
+    border:1px solid #ccc;
+    padding:4px;
+    margin:0px 2px 1px 1px;
+    position:sticky;
+    top:0px;
+    z-index:21;
+}
+
+.jexcel_toolbar:empty
+{
+    display:none;
+}
+
+.jexcel_toolbar i.jexcel_toolbar_item 
+{
+    width:24px;
+    height:24px;
+    padding:4px;
+    cursor:pointer;
+    display:inline-block;
+}
+
+.jexcel_toolbar i.jexcel_toolbar_item:hover 
+{
+    background-color:#ddd;
+}
+
+.jexcel_toolbar select.jexcel_toolbar_item 
+{
+    margin-left:2px;
+    margin-right:2px;
+    display:inline-block;
+    border:0px;
+    background-color:transparent;
+    padding-right:10px;
+}
+
+.jexcel .dragging-left
+{
+    background-repeat: no-repeat;
+    background-position:top 50% left 0px;
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M14 7l-5 5 5 5V7z'/%3E%3Cpath fill='none' d='M24 0v24H0V0h24z'/%3E%3C/svg%3E");
+}
+
+.jexcel .dragging-right
+{
+    background-repeat: no-repeat;
+    background-position:top 50% right 0px;
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M10 17l5-5-5-5v10z'/%3E%3Cpath fill='none' d='M0 24V0h24v24H0z'/%3E%3C/svg%3E");
+}
+
+.jexcel_tabs .jexcel_tab
+{
+    display:none;
+}
+
+.jexcel_tabs .jexcel_tab_link
+{
+    display:inline-block;
+    padding:5px;
+    margin-right:2px;
+    margin-bottom:2px;
+    background-color:#f3f3f3;
+    cursor:pointer;
+}
+
+.jexcel_tabs .jexcel_tab_link.selected
+{
+    background-color:#ddd;
+}
+
+.jexcel_hidden_index > tbody > tr > td:first-child,
+.jexcel_hidden_index > thead > tr > td:first-child, 
+.jexcel_hidden_index > colgroup > col:first-child
+{
+    display:none;
+}
+
+
+
+.jexcel .jrating {
+    display: inline-flex;
+}
+.jexcel .jrating > div {
+    zoom: 0.55;
+}
+
+.jexcel .copying-top {
+    border-top:1px dashed #000;
+}
+
+.jexcel .copying-left {
+    border-left:1px dashed #000;
+}
+
+.jexcel .copying-right {
+    border-right:1px dashed #000;
+}
+
+.jexcel .copying-bottom {
+    border-bottom:1px dashed #000;
+}
+
+.jexcel .jexcel_column_filter {
+    background-repeat: no-repeat;
+    background-position: top 50% right 5px;
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' fill='gray' width='18px' height='18px'%3E%3Cpath d='M10 18h4v-2h-4v2zM3 6v2h18V6H3zm3 7h12v-2H6v2z'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E");
+    text-overflow: ellipsis;
+    overflow: hidden;
+    padding: 0px;
+    padding-left: 6px;
+    padding-right: 20px;
+}
+
+.jexcel thead .jexcel_freezed {
+    left: 0px;
+    z-index: 3 !important;
+    box-shadow: 2px 0px 2px 0.2px #ccc !important;
+    -webkit-box-shadow: 2px 0px 2px 0.2px #ccc !important;
+    -moz-box-shadow: 2px 0px 2px 0.2px #ccc !important;
+}
+
+.jexcel tbody .jexcel_freezed {
+    position: relative;
+    background-color: #fff;
+    box-shadow: 1px 1px 1px 1px #ccc !important;
+    -webkit-box-shadow: 2px 4px 4px 0.1px #ccc !important;
+    -moz-box-shadow: 2px 4px 4px 0.1px #ccc !important;
+}
\ No newline at end of file
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.css.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.css.xml
new file mode 100644
index 0000000000000000000000000000000000000000..f75a3e9e7d952cb070c60564dae7e7311b3509c1
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.css.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>jexcel.css</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>text/css</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.js.js b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.js.js
new file mode 100644
index 0000000000000000000000000000000000000000..2f303e77621aaf41bd5dda5601e69c8e42e9fc77
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.js.js
@@ -0,0 +1,14031 @@
+/**
+ * jExcel v4.2.1
+ *
+ * Modified version for Jexcel editor
+ *
+ * Author: Paul Hodel <paul.hodel@gmail.com>
+ * Website: https://bossanova.uk/jexcel/
+ * Description: Create amazing web based spreadsheets.
+ *
+ * This software is distribute under MIT License
+ */
+
+ if (! jSuites && typeof(require) === 'function') {
+    var jSuites = require('jsuites');
+    require('jsuites/dist/jsuites.css');
+}
+
+;(function (global, factory) {
+    typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
+    typeof define === 'function' && define.amd ? define(factory) :
+    global.jexcel = factory();
+}(this, (function () {
+
+    'use strict';
+
+    // Jexcel core object
+
+    var jexcel = (function(el, options) {
+        // Create jexcel object
+        var obj = {};
+        obj.options = {};
+
+        if (! (el instanceof Element || el instanceof HTMLDocument)) {
+            console.error('JEXCEL: el is not a valid DOM element');
+            return false;
+        } else if (el.tagName == 'TABLE') {
+            if (options = jexcel.createFromTable(el, options)) {
+                var div = document.createElement('div');
+                el.parentNode.insertBefore(div, el);
+                el.remove();
+                el = div;
+            } else {
+                console.error('JEXCEL: el is not a valid DOM element');
+                return false;
+            }
+        }
+
+        // Loading default configuration
+        var defaults = {
+            // External data
+            url:null,
+            // Data
+            data:null,
+            // Copy behavior
+            copyCompatibility:false,
+            root:null,
+            // Rows and columns definitions
+            rows:[],
+            columns:[],
+            // Deprected legacy options
+            colHeaders:[],
+            colWidths:[],
+            colAlignments:[],
+            nestedHeaders:null,
+            // Column width that is used by default
+            defaultColWidth:50,
+            defaultColAlign:'center',
+            // Spare rows and columns
+            minSpareRows:0,
+            minSpareCols:0,
+            // Minimal table dimensions
+            minDimensions:[0,0],
+            // Allow Export
+            allowExport:true,
+            // @type {boolean} - Include the header titles on download
+            includeHeadersOnDownload:false,
+            // @type {boolean} - Include the header titles on copy
+            includeHeadersOnCopy:false,
+            // Allow column sorting
+            columnSorting:true,
+            // Allow column dragging
+            columnDrag:false,
+            // Allow column resizing
+            columnResize:true,
+            // Allow row resizing
+            rowResize:false,
+            // Allow row dragging
+            rowDrag:true,
+            // Allow table edition
+            editable:true,
+            // Allow new rows
+            allowInsertRow:true,
+            // Allow new rows
+            allowManualInsertRow:true,
+            // Allow new columns
+            allowInsertColumn:true,
+            // Allow new rows
+            allowManualInsertColumn:true,
+            // Allow row delete
+            allowDeleteRow:true,
+            // Allow deleting of all rows
+            allowDeletingAllRows:false,
+            // Allow column delete
+            allowDeleteColumn:true,
+            // Allow rename column
+            allowRenameColumn:true,
+            // Allow comments
+            allowComments:false,
+            // Global wrap
+            wordWrap:false,
+            // Image options
+            imageOptions: null,
+            // CSV source
+            csv:null,
+            // Filename
+            csvFileName:'jexcel',
+            // Consider first line as header
+            csvHeaders:true,
+            // Delimiters
+            csvDelimiter:',',
+            // First row as header
+            parseTableFirstRowAsHeader:false,
+            parseTableAutoCellType:false,
+            // Disable corner selection
+            selectionCopy:true,
+            // Merged cells
+            mergeCells:{},
+            // Create toolbar
+            toolbar:null,
+            // Allow search
+            search:false,
+            // Create pagination
+            pagination:false,
+            paginationOptions:null,
+            // Full screen
+            fullscreen:false,
+            // Lazy loading
+            lazyLoading:false,
+            loadingSpin:false,
+            // Table overflow
+            tableOverflow:false,
+            tableHeight:'300px',
+            tableWidth:null,
+            // Meta
+            meta: null,
+            // Style
+            style:null,
+            // Execute formulas
+            parseFormulas:true,
+            autoIncrement:true,
+            autoCasting:true,
+            // Security
+            secureFormulas:true,
+            stripHTML:true,
+            // Filters
+            filters:false,
+            footers:null,
+            // Event handles
+            onundo:null,
+            onredo:null,
+            onload:null,
+            onchange:null,
+            onbeforechange:null,
+            onafterchanges:null,
+            onbeforeinsertrow: null,
+            oninsertrow:null,
+            onbeforeinsertcolumn: null,
+            oninsertcolumn:null,
+            onbeforedeleterow:null,
+            ondeleterow:null,
+            onbeforedeletecolumn:null,
+            ondeletecolumn:null,
+            onmoverow:null,
+            onmovecolumn:null,
+            onresizerow:null,
+            onresizecolumn:null,
+            onsort:null,
+            onselection:null,
+            oncopy:null,
+            onpaste:null,
+            onbeforepaste:null,
+            onmerge:null,
+            onfocus:null,
+            onblur:null,
+            onchangeheader:null,
+            oneditionstart:null,
+            oneditionend:null,
+            onchangestyle:null,
+            onchangemeta:null,
+            onchangepage:null,
+            onbeforesave:null,
+            onsave:null,
+            // Global event dispatcher
+            onevent:null,
+            // Persistance
+            persistance:false,
+            // Customize any cell behavior
+            updateTable:null,
+            // Detach the HTML table when calling updateTable
+            detachForUpdates: false,
+            freezeColumns:null,
+            // Texts
+            text:{
+                noRecordsFound: 'No records found',
+                showingPage: 'Showing page {0} of {1} entries',
+                show: 'Show ',
+                search: 'Search',
+                entries: ' entries',
+                columnName: 'Column name',
+                insertANewColumnBefore: 'Insert a new column before',
+                insertANewColumnAfter: 'Insert a new column after',
+                deleteSelectedColumns: 'Delete selected columns',
+                renameThisColumn: 'Rename this column',
+                orderAscending: 'Order ascending',
+                orderDescending: 'Order descending',
+                insertANewRowBefore: 'Insert a new row before',
+                insertANewRowAfter: 'Insert a new row after',
+                deleteSelectedRows: 'Delete selected rows',
+                editComments: 'Edit comments',
+                addComments: 'Add comments',
+                comments: 'Comments',
+                clearComments: 'Clear comments',
+                copy: 'Copy...',
+                paste: 'Paste...',
+                saveAs: 'Save as...',
+                about: 'About',
+                areYouSureToDeleteTheSelectedRows: 'Are you sure to delete the selected rows?',
+                areYouSureToDeleteTheSelectedColumns: 'Are you sure to delete the selected columns?',
+                thisActionWillDestroyAnyExistingMergedCellsAreYouSure: 'This action will destroy any existing merged cells. Are you sure?',
+                thisActionWillClearYourSearchResultsAreYouSure: 'This action will clear your search results. Are you sure?',
+                thereIsAConflictWithAnotherMergedCell: 'There is a conflict with another merged cell',
+                invalidMergeProperties: 'Invalid merged properties',
+                cellAlreadyMerged: 'Cell already merged',
+                noCellsSelected: 'No cells selected',
+            },
+            // About message
+            about:"jExcel CE Spreadsheet\nVersion 4.2.0\nAuthor: Paul Hodel <paul.hodel@gmail.com>\nWebsite: https://bossanova.uk/jexcel/v3",
+        };
+    
+        // Loading initial configuration from user
+        for (var property in defaults) {
+            if (options && options.hasOwnProperty(property)) {
+                if (property === 'text') {
+                    obj.options[property] = defaults[property];
+                    for (var textKey in options[property]) {
+                        if (options[property].hasOwnProperty(textKey)){
+                            obj.options[property][textKey] = options[property][textKey];
+                        }
+                    }
+                } else {
+                    obj.options[property] = options[property];
+                }
+            } else {
+                obj.options[property] = defaults[property];
+            }
+        }
+
+        // Global elements
+        obj.el = el;
+        obj.corner = null;
+        obj.contextMenu = null;
+        obj.textarea = null;
+        obj.ads = null;
+        obj.content = null;
+        obj.table = null;
+        obj.thead = null;
+        obj.tbody = null;
+        obj.rows = [];
+        obj.results = null;
+        obj.searchInput = null;
+        obj.toolbar = null;
+        obj.pagination = null;
+        obj.pageNumber = null;
+        obj.headerContainer = null;
+        obj.colgroupContainer = null;
+    
+        // Containers
+        obj.headers = [];
+        obj.records = [];
+        obj.history = [];
+        obj.formula = [];
+        obj.colgroup = [];
+        obj.selection = [];
+        obj.highlighted  = [];
+        obj.selectedCell = null;
+        obj.selectedContainer = null;
+        obj.style = [];
+        obj.data = null;
+        obj.filter = null;
+        obj.filters = [];
+
+        // Internal controllers
+        obj.cursor = null;
+        obj.historyIndex = -1;
+        obj.ignoreEvents = false;
+        obj.ignoreHistory = false;
+        obj.edition = null;
+        obj.hashString = null;
+        obj.resizing = null;
+        obj.dragging = null;
+    
+        // Lazy loading
+        if (obj.options.lazyLoading == true && (obj.options.tableOverflow == false && obj.options.fullscreen == false)) {
+            console.error('JEXCEL: The lazyloading only works when tableOverflow = yes or fullscreen = yes');
+            obj.options.lazyLoading = false;
+        }
+        
+        /**
+         * Activate/Disable fullscreen 
+         * use programmatically : table.fullscreen(); or table.fullscreen(true); or table.fullscreen(false);
+         * @Param {boolean} activate
+         */
+        obj.fullscreen = function(activate) {
+            // If activate not defined, get reverse options.fullscreen
+            if (activate == null) {
+                activate = ! obj.options.fullscreen;
+            }
+    
+            // If change
+            if (obj.options.fullscreen != activate) {
+                obj.options.fullscreen = activate;
+    
+                // Test LazyLoading conflict
+                if (activate == true) {
+                    el.classList.add('fullscreen');
+                } else {
+                    el.classList.remove('fullscreen');
+                }
+            } 
+        }
+
+        /**
+         * Trigger events
+         */
+        obj.dispatch = function(event) {
+            // Dispatch events
+            if (! obj.ignoreEvents) {
+                // Call global event
+                if (typeof(obj.options.onevent) == 'function') {
+                    var ret = obj.options.onevent.apply(this, arguments);
+                }
+                // Call specific events
+                if (typeof(obj.options[event]) == 'function') {
+                    var ret = obj.options[event].apply(this, Array.prototype.slice.call(arguments, 1));
+                }
+            }
+
+            // Persistance
+            if (event == 'onafterchanges' && obj.options.persistance) {
+                var url = obj.options.persistance == true ? obj.options.url : obj.options.persistance;
+                var data = obj.prepareJson(arguments[2]);
+                obj.save(url, data);
+            }
+
+            return ret;
+        }
+
+        /**
+         * Prepare the jexcel table
+         * 
+         * @Param config
+         */
+        obj.prepareTable = function() {
+            // Loading initial data from remote sources
+            var results = [];
+    
+            // Number of columns
+            var size = obj.options.columns.length;
+    
+            if (obj.options.data && typeof(obj.options.data[0]) !== 'undefined') {
+                // Data keys
+                var keys = Object.keys(obj.options.data[0]);
+    
+                if (keys.length > size) {
+                    size = keys.length;
+                }
+            }
+    
+            // Minimal dimensions
+            if (obj.options.minDimensions[0] > size) {
+                size = obj.options.minDimensions[0];
+            }
+    
+            // Requests
+            var multiple = [];
+    
+            // Preparations
+            for (var i = 0; i < size; i++) {
+                // Deprected options. You should use only columns
+                if (! obj.options.colHeaders[i]) {
+                    obj.options.colHeaders[i] = '';
+                }
+                if (! obj.options.colWidths[i]) {
+                    obj.options.colWidths[i] = obj.options.defaultColWidth;
+                }
+                if (! obj.options.colAlignments[i]) {
+                    obj.options.colAlignments[i] = obj.options.defaultColAlign;
+                }
+    
+                // Default column description
+                if (! obj.options.columns[i]) {
+                    obj.options.columns[i] = { type:'text' };
+                } else if (! obj.options.columns[i].type) {
+                    obj.options.columns[i].type = 'text';
+                }
+                if (! obj.options.columns[i].name) {
+                    obj.options.columns[i].name = keys && keys[i] ? keys[i] : i;
+                }
+                if (! obj.options.columns[i].source) {
+                    obj.options.columns[i].source = [];
+                }
+                if (! obj.options.columns[i].options) {
+                    obj.options.columns[i].options = [];
+                }
+                if (! obj.options.columns[i].editor) {
+                    obj.options.columns[i].editor = null;
+                }
+                if (! obj.options.columns[i].allowEmpty) {
+                    obj.options.columns[i].allowEmpty = false;
+                }
+                if (! obj.options.columns[i].title) {
+                    obj.options.columns[i].title = obj.options.colHeaders[i] ? obj.options.colHeaders[i] : '';
+                }
+                if (! obj.options.columns[i].width) {
+                    obj.options.columns[i].width = obj.options.colWidths[i] ? obj.options.colWidths[i] : obj.options.defaultColWidth;
+                }
+                if (! obj.options.columns[i].align) {
+                    obj.options.columns[i].align = obj.options.colAlignments[i] ? obj.options.colAlignments[i] : 'center';
+                }
+    
+                // Pre-load initial source for json autocomplete
+                if (obj.options.columns[i].type == 'autocomplete' || obj.options.columns[i].type == 'dropdown') {
+                    // if remote content
+                    if (obj.options.columns[i].url) {
+                        multiple.push(jSuites.ajax({
+                            url: obj.options.columns[i].url,
+                            index: i,
+                            method: 'GET',
+                            dataType: 'json',
+                            multiple: multiple,
+                            success: function(data) {
+                                var source = [];
+                                for (var i = 0; i < data.length; i++) {
+                                    obj.options.columns[this.index].source.push(data[i]);
+                                }
+                            },
+                            complete: function() {
+                                obj.createTable();
+                            }
+                        }));
+                    }
+                } else if (obj.options.columns[i].type == 'calendar') {
+                    // Default format for date columns
+                    if (! obj.options.columns[i].options.format) {
+                        obj.options.columns[i].options.format = 'DD/MM/YYYY';
+                    }
+                }
+            }
+    
+            // On complete
+            if (! multiple.length) {
+                obj.createTable();
+            }
+        }
+    
+        obj.createTable = function() {
+            // Elements
+            obj.table = document.createElement('table');
+            obj.thead = document.createElement('thead');
+            obj.tbody = document.createElement('tbody');
+
+            // Create headers controllers
+            obj.headers = [];
+            obj.colgroup = [];
+    
+            // Create table container
+            obj.content = document.createElement('div');
+            obj.content.classList.add('jexcel_content');
+            obj.content.onscroll = function(e) {
+                obj.scrollControls(e);
+            }
+            obj.content.onwheel = function(e) {
+                obj.wheelControls(e);
+            }
+
+            // Create toolbar object
+            obj.toolbar = document.createElement('div');
+            obj.toolbar.classList.add('jexcel_toolbar');
+    
+            // Search
+            var searchContainer = document.createElement('div');
+            var searchText = document.createTextNode((obj.options.text.search) + ': ');
+            obj.searchInput = document.createElement('input');
+            obj.searchInput.classList.add('jexcel_search');
+            searchContainer.appendChild(searchText);
+            searchContainer.appendChild(obj.searchInput);
+            obj.searchInput.onfocus = function() {
+                obj.resetSelection();
+            }
+    
+            // Pagination select option
+            var paginationUpdateContainer = document.createElement('div');
+    
+            if (obj.options.pagination > 0 && obj.options.paginationOptions && obj.options.paginationOptions.length > 0) {
+                obj.paginationDropdown = document.createElement('select');
+                obj.paginationDropdown.classList.add('jexcel_pagination_dropdown');
+                obj.paginationDropdown.onchange = function() {
+                    obj.options.pagination = parseInt(this.value);
+                    obj.page(0);
+                }
+    
+                for (var i = 0; i < obj.options.paginationOptions.length; i++) {
+                    var temp = document.createElement('option');
+                    temp.value = obj.options.paginationOptions[i];
+                    temp.innerHTML = obj.options.paginationOptions[i];
+                    obj.paginationDropdown.appendChild(temp);
+                }
+    
+                paginationUpdateContainer.appendChild(document.createTextNode(obj.options.text.show));
+                paginationUpdateContainer.appendChild(obj.paginationDropdown);
+                paginationUpdateContainer.appendChild(document.createTextNode(obj.options.text.entries));
+            }
+    
+            // Filter and pagination container
+            var filter = document.createElement('div');
+            filter.classList.add('jexcel_filter');
+            filter.appendChild(paginationUpdateContainer);
+            filter.appendChild(searchContainer);
+    
+            // Colsgroup
+            obj.colgroupContainer = document.createElement('colgroup');
+            var tempCol = document.createElement('col');
+            tempCol.setAttribute('width', '50');
+            obj.colgroupContainer.appendChild(tempCol);
+    
+            // Nested
+            if (obj.options.nestedHeaders && obj.options.nestedHeaders.length > 0) {
+                // Flexible way to handle nestedheaders
+                if (obj.options.nestedHeaders[0] && obj.options.nestedHeaders[0][0]) {
+                    for (var j = 0; j < obj.options.nestedHeaders.length; j++) {
+                        obj.thead.appendChild(obj.createNestedHeader(obj.options.nestedHeaders[j]));
+                    }
+                } else {
+                    obj.thead.appendChild(obj.createNestedHeader(obj.options.nestedHeaders));
+                }
+            }
+    
+            // Row
+            obj.headerContainer = document.createElement('tr');
+            var tempCol = document.createElement('td');
+            tempCol.classList.add('jexcel_selectall');
+            obj.headerContainer.appendChild(tempCol);
+    
+            for (var i = 0; i < obj.options.columns.length; i++) {
+                // Create header
+                obj.createCellHeader(i);
+                // Append cell to the container
+                obj.headerContainer.appendChild(obj.headers[i]);
+                obj.colgroupContainer.appendChild(obj.colgroup[i]);
+            }
+
+            obj.thead.appendChild(obj.headerContainer);
+
+            // Filters
+            if (obj.options.filters == true) {
+                obj.filter = document.createElement('tr');
+                var td = document.createElement('td');
+                obj.filter.appendChild(td);
+
+                for (var i = 0; i < obj.options.columns.length; i++) {
+                    var td = document.createElement('td');
+                    td.innerHTML = '&nbsp;';
+                    td.setAttribute('data-x', i);
+                    td.className = 'jexcel_column_filter';
+                    obj.filter.appendChild(td);
+                }
+
+                obj.thead.appendChild(obj.filter);
+            }
+
+            // Content table
+            obj.table = document.createElement('table');
+            obj.table.classList.add('jexcel');
+            obj.table.setAttribute('cellpadding', '0');
+            obj.table.setAttribute('cellspacing', '0');
+            obj.table.setAttribute('unselectable', 'yes');
+            //obj.table.setAttribute('onselectstart', 'return false');
+            obj.table.appendChild(obj.colgroupContainer);
+            obj.table.appendChild(obj.thead);
+            obj.table.appendChild(obj.tbody);
+
+            // Spreadsheet corner
+            obj.corner = document.createElement('div');
+            obj.corner.className = 'jexcel_corner';
+            obj.corner.setAttribute('unselectable', 'on');
+            obj.corner.setAttribute('onselectstart', 'return false');
+    
+            if (obj.options.selectionCopy == false) {
+                obj.corner.style.display = 'none';
+            }
+    
+            // Textarea helper
+            obj.textarea = document.createElement('textarea');
+            obj.textarea.className = 'jexcel_textarea';
+            obj.textarea.id = 'jexcel_textarea';
+            obj.textarea.tabIndex = '-1';
+
+            // Contextmenu container
+            obj.contextMenu = document.createElement('div');
+            obj.contextMenu.className = 'jexcel_contextmenu';
+    
+            // Create element
+            jSuites.contextmenu(obj.contextMenu, {
+                onclick:function() {
+                    obj.contextMenu.contextmenu.close(false);
+                }
+            });
+    
+            // Powered by jExcel
+            var ads = document.createElement('a');
+            ads.setAttribute('href', 'https://bossanova.uk/jexcel/');
+            obj.ads = document.createElement('div');
+            obj.ads.className = 'jexcel_about';
+            try {
+                if (typeof(sessionStorage) !== "undefined" && ! sessionStorage.getItem('jexcel')) {
+                    sessionStorage.setItem('jexcel', true);
+                    var img = document.createElement('img');
+                    img.src = '//bossanova.uk/jexcel/logo.png';
+                    ads.appendChild(img);
+                }
+            } catch (exception) {
+            }
+            var span = document.createElement('span');
+            span.innerHTML = 'Jexcel spreadsheet';
+            ads.appendChild(span);
+            obj.ads.appendChild(ads);
+
+            // Create table container TODO: frozen columns
+            var container = document.createElement('div');
+            container.classList.add('jexcel_table');
+    
+            // Pagination
+            obj.pagination = document.createElement('div');
+            obj.pagination.classList.add('jexcel_pagination');
+            var paginationInfo = document.createElement('div');
+            var paginationPages = document.createElement('div');
+            obj.pagination.appendChild(paginationInfo);
+            obj.pagination.appendChild(paginationPages);
+
+            // Hide pagination if not in use
+            if (! obj.options.pagination) {
+                obj.pagination.style.display = 'none';
+            }
+
+            // Append containers to the table
+            if (obj.options.search == true) {
+                el.appendChild(filter);
+            }
+    
+            // Elements
+            obj.content.appendChild(obj.table);
+            obj.content.appendChild(obj.corner);
+            obj.content.appendChild(obj.textarea);
+    
+            el.appendChild(obj.toolbar);
+            el.appendChild(obj.content);
+            el.appendChild(obj.pagination);
+            el.appendChild(obj.contextMenu);
+            el.appendChild(obj.ads);
+            el.classList.add('jexcel_container');
+    
+            // Create toolbar
+            if (obj.options.toolbar && obj.options.toolbar.length) {
+                obj.createToolbar();
+            }
+    
+            // Fullscreen
+            if (obj.options.fullscreen == true) {
+                el.classList.add('fullscreen');
+            } else {
+                // Overflow
+                if (obj.options.tableOverflow == true) {
+                    if (obj.options.tableHeight) {
+                        obj.content.style['overflow-y'] = 'auto';
+                        obj.content.style.maxHeight = obj.options.tableHeight;
+                    }
+                    if (obj.options.tableWidth) {
+                        obj.content.style['overflow-x'] = 'auto';
+                        obj.content.style.width = obj.options.tableWidth;
+                    }
+                }
+            }
+    
+            // With toolbars
+            if (obj.options.tableOverflow != true && obj.options.toolbar) {
+                el.classList.add('with-toolbar');
+            }
+    
+            // Actions
+            if (obj.options.columnDrag == true) {
+                obj.thead.classList.add('draggable');
+            }
+            if (obj.options.columnResize == true) {
+                obj.thead.classList.add('resizable');
+            }
+            if (obj.options.rowDrag == true) {
+                obj.tbody.classList.add('draggable');
+            }
+            if (obj.options.rowResize == true) {
+                obj.tbody.classList.add('resizable');
+            }
+    
+            // Load data
+            obj.setData();
+    
+            // Style
+            if (obj.options.style) {
+                obj.setStyle(obj.options.style, null, null, 1, 1);
+            }
+        }
+
+        /**
+         * Refresh the data
+         * 
+         * @return void
+         */
+        obj.refresh = function() {
+            if (obj.options.url) {
+                // Loading
+                if (obj.options.loadingSpin == true) {
+                    jSuites.loading.show();
+                }
+    
+                jSuites.ajax({
+                    url: obj.options.url,
+                    method: 'GET',
+                    dataType: 'json',
+                    success: function(result) {
+                        // Data
+                        obj.options.data = (result.data) ? result.data : result;
+                        // Prepare table
+                        obj.setData();
+                        // Hide spin
+                        if (obj.options.loadingSpin == true) {
+                            jSuites.loading.hide();
+                        }
+                    }
+                });
+            } else {
+                obj.setData();
+            }
+        }
+
+        /**
+         * Set data
+         * 
+         * @param array data In case no data is sent, default is reloaded
+         * @return void
+         */
+        obj.setData = function(data) {
+            // Update data
+            if (data) {
+                if (typeof(data) == 'string') {
+                    data = JSON.parse(data);
+                }
+    
+                obj.options.data = data;
+            }
+    
+            // Data
+            if (! obj.options.data) {
+                obj.options.data = [];
+            }
+    
+            // Prepare data
+            if (obj.options.data && obj.options.data[0]) {
+                if (! Array.isArray(obj.options.data[0])) {
+                    var data = [];
+                    for (var j = 0; j < obj.options.data.length; j++) {
+                        var row = [];
+                        for (var i = 0; i < obj.options.columns.length; i++) {
+                            row[i] = obj.options.data[j][obj.options.columns[i].name];
+                        }
+                        data.push(row);
+                    }
+
+                    obj.options.data = data;
+                }
+            }
+
+            // Adjust minimal dimensions
+            var j = 0;
+            var i = 0;
+            var size_i = obj.options.columns.length;
+            var size_j = obj.options.data.length;
+            var min_i = obj.options.minDimensions[0];
+            var min_j = obj.options.minDimensions[1];
+            var max_i = min_i > size_i ? min_i : size_i;
+            var max_j = min_j > size_j ? min_j : size_j;
+    
+            for (j = 0; j < max_j; j++) {
+                for (i = 0; i < max_i; i++) {
+                    if (obj.options.data[j] == undefined) {
+                        obj.options.data[j] = [];
+                    }
+    
+                    if (obj.options.data[j][i] == undefined) {
+                        obj.options.data[j][i] = '';
+                    }
+                }
+            }
+    
+            // Reset containers
+            obj.rows = [];
+            obj.results = null;
+            obj.records = [];
+            obj.history = [];
+    
+            // Reset internal controllers
+            obj.historyIndex = -1;
+    
+            // Reset data
+            obj.tbody.innerHTML = '';
+    
+            // Lazy loading
+            if (obj.options.lazyLoading == true) {
+                // Load only 100 records
+                var startNumber = 0
+                var finalNumber = obj.options.data.length < 100 ? obj.options.data.length : 100;
+    
+                if (obj.options.pagination) {
+                    obj.options.pagination = false;
+                    console.error('JEXCEL: Pagination will be disable due the lazyLoading');
+                }
+            } else if (obj.options.pagination) {
+                // Pagination
+                if (! obj.pageNumber) {
+                    obj.pageNumber = 0;
+                }
+                var quantityPerPage = obj.options.pagination;
+                startNumber = (obj.options.pagination * obj.pageNumber);
+                finalNumber = (obj.options.pagination * obj.pageNumber) + obj.options.pagination;
+    
+                if (obj.options.data.length < finalNumber) {
+                    finalNumber = obj.options.data.length;
+                }
+            } else {
+                var startNumber = 0;
+                var finalNumber = obj.options.data.length;
+            }
+    
+            // Append nodes to the HTML
+            for (j = 0; j < obj.options.data.length; j++) {
+                // Create row
+                var tr = obj.createRow(j, obj.options.data[j]);
+                // Append line to the table
+                if (j >= startNumber && j < finalNumber) {
+                    obj.tbody.appendChild(tr);
+                }
+            }
+    
+            if (obj.options.lazyLoading == true) {
+                // Do not create pagination with lazyloading activated
+            } else if (obj.options.pagination) {
+                obj.updatePagination();
+            }
+    
+            // Merge cells
+            if (obj.options.mergeCells) {
+                var keys = Object.keys(obj.options.mergeCells);
+                for (var i = 0; i < keys.length; i++) {
+                    var num = obj.options.mergeCells[keys[i]];
+                    obj.setMerge(keys[i], num[0], num[1], 1);
+                }
+            }
+
+            // Updata table with custom configurations if applicable
+            obj.updateTable();
+
+            // Onload
+            obj.dispatch('onload', el, obj);
+        }
+
+        /**
+         * Get the whole table data
+         * 
+         * @param bool get highlighted cells only
+         * @return array data
+         */
+        obj.getData = function(highlighted, dataOnly) {
+            // Control vars
+            var dataset = [];
+            var px = 0;
+            var py = 0;
+
+            // Data type
+            var dataType = dataOnly == true || obj.options.copyCompatibility == false ? true : false;
+
+            // Column and row length
+            var x = obj.options.columns.length
+            var y = obj.options.data.length
+    
+            // Go through the columns to get the data
+            for (var j = 0; j < y; j++) {
+                px = 0;
+                for (var i = 0; i < x; i++) {
+                    // Cell selected or fullset
+                    if (! highlighted || obj.records[j][i].classList.contains('highlight')) {
+                        // Get value
+                        if (! dataset[py]) {
+                            dataset[py] = [];
+                        }
+                        if (! dataType) {
+                            dataset[py][px] = obj.records[j][i].innerHTML;
+                        } else {
+                            dataset[py][px] = obj.options.data[j][i];
+                        }
+                        px++;
+                    }
+                }
+                if (px > 0) {
+                    py++;
+                }
+           }
+    
+           return dataset;
+        }
+    
+        /**
+         * Get the whole table data
+         * 
+         * @param integer row number
+         * @return string value
+         */
+        obj.getJson = function(highlighted) {
+            // Control vars
+            var data = [];
+    
+            // Column and row length
+            var x = obj.options.columns.length
+            var y = obj.options.data.length
+    
+            // Go through the columns to get the data
+            for (var j = 0; j < y; j++) {
+                var row = null;
+                for (var i = 0; i < x; i++) {
+                    if (! highlighted || obj.records[j][i].classList.contains('highlight')) {
+                        if (row == null) {
+                            row = {};
+                        }
+                        if (! obj.options.columns[i].name) {
+                            obj.options.columns[i].name = i;
+                        }
+                        row[obj.options.columns[i].name] = obj.options.data[j][i];
+                    }
+                }
+    
+                if (row != null) {
+                    data.push(row);
+                }
+           }
+    
+           return data;
+        }
+
+        /**
+         * Prepare JSON in the correct format
+         */
+        obj.prepareJson = function(data) {
+            var rows = [];
+            for (var i = 0; i < data.length; i++) {
+                var x = data[i].x;
+                var y = data[i].y;
+                var k = obj.options.columns[x].name ? obj.options.columns[x].name : x;
+
+                // Create row
+                if (! rows[y]) {
+                    rows[y] = {
+                        row: y,
+                        data: {},
+                    };
+                }
+                rows[y].data[k] = data[i].newValue;
+            }
+
+            // Filter rows
+            return rows.filter(function (el) {
+                return el != null;
+            });
+        }
+
+        /**
+         * Post json to a remote server
+         */
+        obj.save = function(url, data) {
+            // Parse anything in the data before sending to the server
+            var ret = obj.dispatch('onbeforesave', el, obj, data);
+console.log(ret);
+            if (ret) {
+                var data = ret;
+            } else {
+                if (ret === false) {
+                    return false;
+                }
+            }
+
+            // Remove update
+            jSuites.ajax({
+                url: url,
+                method: 'POST',
+                dataType: 'json',
+                data: { data: JSON.stringify(data) },
+                success: function(result) {
+                    // Event
+                    obj.dispatch('onsave', el, obj, data);
+                }
+            });
+        }
+
+        /**
+         * Get a row data by rowNumber
+         */
+        obj.getRowData = function(rowNumber) {
+            return obj.options.data[rowNumber];
+        }
+    
+        /**
+         * Set a row data by rowNumber
+         */
+        obj.setRowData = function(rowNumber, data) {
+            for (var i = 0; i < obj.headers.length; i++) {
+                // Update cell
+                var columnName = jexcel.getColumnNameFromId([ i, rowNumber ]);
+                // Set value
+                if (data[i] != null) {
+                    obj.setValue(columnName, data[i]);
+                }
+            }
+        }
+    
+        /**
+         * Get a column data by columnNumber
+         */
+        obj.getColumnData = function(columnNumber) {
+            var dataset = [];
+            // Go through the rows to get the data
+            for (var j = 0; j < obj.options.data.length; j++) {
+                dataset.push(obj.options.data[j][columnNumber]);
+            }
+            return dataset;
+        }
+    
+        /**
+         * Set a column data by colNumber
+         */
+        obj.setColumnData = function(colNumber, data) {
+            for (var j = 0; j < obj.rows.length; j++) {
+                // Update cell
+                var columnName = jexcel.getColumnNameFromId([ colNumber, j ]);
+                // Set value
+                if (data[j] != null) {
+                    obj.setValue(columnName, data[j]);
+                }
+            }
+        }
+    
+        /**
+         * Create row
+         */
+        obj.createRow = function(j, data) {
+            // Create container
+            if (! obj.records[j]) {
+                obj.records[j] = [];
+            }
+            // Default data
+            if (! data) {
+                var data = obj.options.data[j];
+            }
+            // New line of data to be append in the table
+            obj.rows[j] = document.createElement('tr');
+            obj.rows[j].setAttribute('data-y', j);
+            // Definitions
+            if (obj.options.rows[j]) {
+                if (obj.options.rows[j].height) {
+                    obj.rows[j].style.height = obj.options.rows[j].height;
+                }
+            }
+            // Row number label
+            var td = document.createElement('td');
+            td.innerHTML = parseInt(j + 1);
+            td.setAttribute('data-y', j);
+            td.className = 'jexcel_row';
+            obj.rows[j].appendChild(td);
+    
+            // Data columns
+            for (var i = 0; i < obj.options.columns.length; i++) {
+                // New column of data to be append in the line
+                obj.records[j][i] = obj.createCell(i, j, data[i]);
+                // Add column to the row
+                obj.rows[j].appendChild(obj.records[j][i]);
+            }
+    
+            // Add row to the table body
+            return obj.rows[j];
+        }
+    
+        obj.parseValue = function(i, j, value) {
+            if ((''+value).substr(0,1) == '=' && obj.options.parseFormulas == true) {
+                value = obj.executeFormula(value, i, j)
+            }
+            if (obj.options.columns[i].mask) {
+                var decimal = obj.options.columns[i].decimal || '.';
+                value = '' + jSuites.mask.run(value, obj.options.columns[i].mask, decimal);
+            }
+
+            return value;
+        }
+
+        /**
+         * Create cell
+         */
+        obj.createCell = function(i, j, value) {
+            // Create cell and properties
+            var td = document.createElement('td');
+            td.setAttribute('data-x', i);
+            td.setAttribute('data-y', j);
+
+            // Security
+            if ((''+value).substr(0,1) == '=' && obj.options.secureFormulas == true) {
+                var val = secureFormula(value);
+                if (val != value) {
+                    // Update the data container
+                    value = val;
+                }
+            }
+
+            // Custom column
+            if (obj.options.columns[i].editor) {
+                if (obj.options.stripHTML === false || obj.options.columns[i].stripHTML === false) {
+                    td.innerHTML =  value;
+                } else {
+                    td.innerText = value;
+                }
+                if (typeof(obj.options.columns[i].editor.createCell) == 'function') {
+                    td = obj.options.columns[i].editor.createCell(td);
+                }
+            } else {
+                // Hidden column
+                if (obj.options.columns[i].type == 'hidden') {
+                    td.style.display = 'none';
+                    td.innerText = value;
+                } else if (obj.options.columns[i].type == 'checkbox' || obj.options.columns[i].type == 'radio') {
+                    // Create input
+                    var element = document.createElement('input');
+                    element.type = obj.options.columns[i].type;
+                    element.name = 'c' + i;
+                    element.checked = (value == 1 || value == true || value == 'true') ? true : false;
+                    element.onclick = function() {
+                        obj.setValue(td, this.checked);
+                    }
+
+                    if (obj.options.columns[i].readOnly == true || obj.options.editable == false) {
+                        element.setAttribute('disabled', 'disabled');
+                    }
+
+                    // Append to the table
+                    td.appendChild(element);
+                    // Make sure the values are correct
+                    obj.options.data[j][i] = element.checked;
+                } else if (obj.options.columns[i].type == 'calendar') {
+                    // Try formatted date
+                    var formatted = jSuites.calendar.extractDateFromString(value, obj.options.columns[i].options.format);
+                    // Create calendar cell
+                    td.innerText = jSuites.calendar.getDateString(formatted ? formatted : value, obj.options.columns[i].options.format);
+                } else if (obj.options.columns[i].type == 'dropdown' || obj.options.columns[i].type == 'autocomplete') {
+                    // Create dropdown cell
+                    td.classList.add('jexcel_dropdown');
+                    td.innerText = obj.getDropDownValue(i, value);
+                } else if (obj.options.columns[i].type == 'color') {
+                    if (obj.options.columns[i].render == 'square') {
+                        var color = document.createElement('div');
+                        color.className = 'color';
+                        color.style.backgroundColor = value;
+                        td.appendChild(color);
+                    } else {
+                        td.style.color = value;
+                        td.innerText = value;
+                    }
+                } else if (obj.options.columns[i].type == 'image') {
+                    if (value && value.substr(0, 10) == 'data:image') {
+                        var img = document.createElement('img');
+                        img.src = value;
+                        td.appendChild(img);
+                    }
+                } else {
+                    if (obj.options.columns[i].type == 'html') {
+                        td.innerHTML = stripScript(obj.parseValue(i, j, value));
+                    } else {
+                        if (obj.options.stripHTML === false || obj.options.columns[i].stripHTML === false) {
+                            td.innerHTML = stripScript(obj.parseValue(i, j, value));
+                        } else {
+                            td.innerText = obj.parseValue(i, j, value);
+                        }
+                    }
+                }
+            }
+    
+            // Readonly
+            if (obj.options.columns[i].readOnly == true) {
+                td.className = 'readonly';
+            }
+    
+            // Text align
+            var colAlign = obj.options.columns[i].align ? obj.options.columns[i].align : 'center';
+            td.style.textAlign = colAlign;
+    
+            // Wrap option
+            if (obj.options.columns[i].wordWrap != false && (obj.options.wordWrap == true || obj.options.columns[i].wordWrap == true || td.innerHTML.length > 200)) {
+                td.style.whiteSpace = 'pre-wrap';
+            }
+    
+            // Overflow
+            if (i > 0) {
+                if (value || td.innerHTML) {
+                    obj.records[j][i-1].style.overflow = 'hidden';
+                } else {
+                    if (i == obj.options.columns.length - 1) {
+                        td.style.overflow = 'hidden';
+                    }
+                }
+            }
+    
+            return td;
+        }
+    
+        obj.createCellHeader = function(colNumber) {
+            // Create col global control
+            var colWidth = obj.options.columns[colNumber].width ? obj.options.columns[colNumber].width : obj.options.defaultColWidth;
+            var colAlign = obj.options.columns[colNumber].align ? obj.options.columns[colNumber].align : obj.options.defaultColAlign;
+    
+            // Create header cell
+            obj.headers[colNumber] = document.createElement('td');
+            obj.headers[colNumber].innerText = obj.options.columns[colNumber].title ? obj.options.columns[colNumber].title : jexcel.getColumnName(colNumber);
+            obj.headers[colNumber].setAttribute('data-x', colNumber);
+            obj.headers[colNumber].style.textAlign = colAlign;
+            if (obj.options.columns[colNumber].title) {
+                obj.headers[colNumber].setAttribute('title', obj.options.columns[colNumber].title);
+            }
+    
+            // Width control
+            obj.colgroup[colNumber] = document.createElement('col');
+            obj.colgroup[colNumber].setAttribute('width', colWidth);
+    
+            // Hidden column
+            if (obj.options.columns[colNumber].type == 'hidden') {
+                obj.headers[colNumber].style.display = 'none';
+                obj.colgroup[colNumber].style.display = 'none';
+            }
+        }
+
+        /**
+         * Update a nested header title
+         */
+        obj.updateNestedHeader = function(x, y, title) {
+            if (obj.options.nestedHeaders[y][x].title) {
+                obj.options.nestedHeaders[y][x].title = title;
+                obj.options.nestedHeaders[y].element.children[x+1].innerText = title;
+            }
+        }
+
+        /**
+         * Create a nested header object
+         */
+        obj.createNestedHeader = function(nestedInformation) {
+            var tr = document.createElement('tr');
+            tr.classList.add('jexcel_nested');
+            var td = document.createElement('td');
+            tr.appendChild(td);
+            // Element
+            nestedInformation.element = tr;
+    
+            var headerIndex = 0;
+            for (var i = 0; i < nestedInformation.length; i++) {
+                // Default values
+                if (! nestedInformation[i].colspan) {
+                    nestedInformation[i].colspan = 1;
+                }
+                if (! nestedInformation[i].align) {
+                    nestedInformation[i].align = 'center';
+                }
+                if (! nestedInformation[i].title) {
+                    nestedInformation[i].title = '';
+                }
+    
+                // Number of columns
+                var numberOfColumns = nestedInformation[i].colspan;
+    
+                // Classes container
+                var column = [];
+                // Header classes for this cell
+                for (var x = 0; x < numberOfColumns; x++) {
+                    if (obj.options.columns[headerIndex] && obj.options.columns[headerIndex].type == 'hidden') {
+                        numberOfColumns++;
+                    }
+                    column.push(headerIndex);
+                    headerIndex++;
+                }
+    
+                // Created the nested cell
+                var td = document.createElement('td');
+                td.setAttribute('data-column', column.join(','));
+                td.setAttribute('colspan', nestedInformation[i].colspan);
+                td.setAttribute('align', nestedInformation[i].align);
+                td.innerText = nestedInformation[i].title;
+                tr.appendChild(td);
+            }
+    
+            return tr;
+        }
+    
+        /**
+         * Create toolbar
+         */
+        obj.createToolbar = function(toolbar) {
+            if (toolbar) {
+                obj.options.toolbar = toolbar;
+            } else {
+                var toolbar = obj.options.toolbar;
+            }
+    
+            for (var i = 0; i < toolbar.length; i++) {
+                if (toolbar[i].type == 'i') {
+                    var toolbarItem = document.createElement('i');
+                    toolbarItem.classList.add('jexcel_toolbar_item');
+                    toolbarItem.classList.add('material-icons');
+                    toolbarItem.setAttribute('data-k', toolbar[i].k);
+                    toolbarItem.setAttribute('data-v', toolbar[i].v);
+                    // Tooltip
+                    if (toolbar[i].tooltip) {
+                        toolbarItem.setAttribute('title', toolbar[i].tooltip);
+                    }
+                    // Handle click
+                    if (toolbar[i].onclick && typeof(toolbar[i].onclick)) {
+                        toolbarItem.onclick = (function (a) {
+                            var b = a;
+                            return function () {
+                                toolbar[b].onclick(el, obj, this);
+                            };
+                        })(i);
+                    } else {
+                        toolbarItem.onclick = function() {
+                            var k = this.getAttribute('data-k');
+                            var v = this.getAttribute('data-v');
+                            obj.setStyle(obj.highlighted, k, v);
+                        }
+                    }
+                    // Append element
+                    toolbarItem.innerText = toolbar[i].content;
+                    obj.toolbar.appendChild(toolbarItem);
+                } else if (toolbar[i].type == 'select') {
+                   var toolbarItem = document.createElement('select');
+                   toolbarItem.classList.add('jexcel_toolbar_item');
+                   toolbarItem.setAttribute('data-k', toolbar[i].k);
+                   // Tooltip
+                   if (toolbar[i].tooltip) {
+                       toolbarItem.setAttribute('title', toolbar[i].tooltip);
+                   }
+                   // Handle onchange
+                   if (toolbar[i].onchange && typeof(toolbar[i].onchange)) {
+                       toolbarItem.onchange = toolbar[i].onchange;
+                   } else {
+                       toolbarItem.onchange = function() {
+                           var k = this.getAttribute('data-k');
+                           obj.setStyle(obj.highlighted, k, this.value);
+                       }
+                   }
+                   // Add options to the dropdown
+                   for(var j = 0; j < toolbar[i].v.length; j++) {
+                        var toolbarDropdownOption = document.createElement('option');
+                        toolbarDropdownOption.value = toolbar[i].v[j];
+                        toolbarDropdownOption.innerText = toolbar[i].v[j];
+                        toolbarItem.appendChild(toolbarDropdownOption);
+                   }
+                   obj.toolbar.appendChild(toolbarItem);
+                } else if (toolbar[i].type == 'color') {
+                     var toolbarItem = document.createElement('i');
+                     toolbarItem.classList.add('jexcel_toolbar_item');
+                     toolbarItem.classList.add('material-icons');
+                     toolbarItem.setAttribute('data-k', toolbar[i].k);
+                     toolbarItem.setAttribute('data-v', '');
+                     // Tooltip
+                     if (toolbar[i].tooltip) {
+                         toolbarItem.setAttribute('title', toolbar[i].tooltip);
+                     }
+                     obj.toolbar.appendChild(toolbarItem);
+                     toolbarItem.onclick = function() {
+                         this.color.open();
+                     }
+                     toolbarItem.innerText = toolbar[i].content;
+                     jSuites.color(toolbarItem, {
+                         onchange:function(o, v) {
+                             var k = o.getAttribute('data-k');
+                             obj.setStyle(obj.highlighted, k, v);
+                         }
+                     });
+                }
+            }
+        }
+    
+        /**
+         * Merge cells
+         * @param cellName
+         * @param colspan
+         * @param rowspan
+         * @param ignoreHistoryAndEvents
+         */
+        obj.setMerge = function(cellName, colspan, rowspan, ignoreHistoryAndEvents) {
+            var test = false;
+    
+            if (! cellName) {
+                if (! obj.highlighted.length) {
+                    alert(obj.options.text.noCellsSelected);
+                    return null;
+                } else {
+                    var x1 = parseInt(obj.highlighted[0].getAttribute('data-x'));
+                    var y1 = parseInt(obj.highlighted[0].getAttribute('data-y'));
+                    var x2 = parseInt(obj.highlighted[obj.highlighted.length-1].getAttribute('data-x'));
+                    var y2 = parseInt(obj.highlighted[obj.highlighted.length-1].getAttribute('data-y'));
+                    var cellName = jexcel.getColumnNameFromId([ x1, y1 ]);
+                    var colspan = (x2 - x1) + 1;
+                    var rowspan = (y2 - y1) + 1;
+                }
+            }
+    
+            var cell = jexcel.getIdFromColumnName(cellName, true);
+    
+            if (obj.options.mergeCells[cellName]) {
+                if (obj.records[cell[1]][cell[0]].getAttribute('data-merged')) {
+                    test = obj.options.text.cellAlreadyMerged;
+                }
+            } else if ((! colspan || colspan < 2) && (! rowspan || rowspan < 2)) {
+                test = obj.options.text.invalidMergeProperties;
+            } else {
+                var cells = [];
+                for (var j = cell[1]; j < cell[1] + rowspan; j++) {
+                    for (var i = cell[0]; i < cell[0] + colspan; i++) {
+                        var columnName = jexcel.getColumnNameFromId([i, j]);
+                        if (obj.records[j][i].getAttribute('data-merged')) {
+                            test = obj.options.text.thereIsAConflictWithAnotherMergedCell;
+                        }
+                    }
+                }
+            }
+    
+            if (test) {
+                alert(test);
+            } else {
+                // Add property
+                if (colspan > 1) {
+                    obj.records[cell[1]][cell[0]].setAttribute('colspan', colspan);
+                } else {
+                    colspan = 1;
+                }
+                if (rowspan > 1) {
+                    obj.records[cell[1]][cell[0]].setAttribute('rowspan', rowspan);
+                } else {
+                    rowspan = 1;
+                }
+                // Keep links to the existing nodes
+                obj.options.mergeCells[cellName] = [ colspan, rowspan, [] ];
+                // Mark cell as merged
+                obj.records[cell[1]][cell[0]].setAttribute('data-merged', 'true');
+                // Overflow
+                obj.records[cell[1]][cell[0]].style.overflow = 'hidden';
+                // History data
+                var data = [];
+                // Adjust the nodes
+                for (var y = cell[1]; y < cell[1] + rowspan; y++) {
+                    for (var x = cell[0]; x < cell[0] + colspan; x++) {
+                        if (! (cell[0] == x && cell[1] == y)) {
+                            data.push(obj.options.data[y][x]);
+                            obj.updateCell(x, y, '', true);
+                            obj.options.mergeCells[cellName][2].push(obj.records[y][x]);
+                            obj.records[y][x].style.display = 'none';
+                            obj.records[y][x] = obj.records[cell[1]][cell[0]];
+                        }
+                    }
+                }
+                // In the initialization is not necessary keep the history
+                //obj.updateSelection(obj.records[cell[1]][cell[0]]);
+    
+                if (! ignoreHistoryAndEvents) {
+                    obj.setHistory({
+                        action:'setMerge',
+                        column:cellName,
+                        colspan:colspan,
+                        rowspan:rowspan,
+                        data:data,
+                    });
+    
+                    obj.dispatch('onmerge', el, cellName, colspan, rowspan);
+                }
+            }
+        }
+    
+        /**
+         * Merge cells
+         * @param cellName
+         * @param colspan
+         * @param rowspan
+         * @param ignoreHistoryAndEvents
+         */
+        obj.getMerge = function(cellName) {
+            var data = {};
+            if (cellName) {
+                if (obj.options.mergeCells[cellName]) {
+                    data = [ obj.options.mergeCells[cellName][0], obj.options.mergeCells[cellName][1] ];
+                } else {
+                    data = null;
+                }
+            } else {
+                if (obj.options.mergeCells) {
+                    var mergedCells = obj.options.mergeCells;
+                    var keys = Object.keys(obj.options.mergeCells);
+                    for (var i = 0; i < keys.length; i++) {
+                        data[keys[i]] = [ obj.options.mergeCells[keys[i]][0], obj.options.mergeCells[keys[i]][1] ];
+                    }
+                }
+            }
+    
+            return data;
+        }
+    
+        /**
+         * Remove merge by cellname
+         * @param cellName
+         */
+        obj.removeMerge = function(cellName, data, keepOptions) {
+            if (obj.options.mergeCells[cellName]) {
+                var cell = jexcel.getIdFromColumnName(cellName, true);
+                obj.records[cell[1]][cell[0]].removeAttribute('colspan');
+                obj.records[cell[1]][cell[0]].removeAttribute('rowspan');
+                obj.records[cell[1]][cell[0]].removeAttribute('data-merged');
+                var info = obj.options.mergeCells[cellName];
+    
+                var index = 0;
+                for (var j = 0; j < info[1]; j++) {
+                    for (var i = 0; i < info[0]; i++) {
+                        if (j > 0 || i > 0) {
+                            obj.records[cell[1]+j][cell[0]+i] = info[2][index];
+                            obj.records[cell[1]+j][cell[0]+i].style.display = '';
+                            // Recover data
+                            if (data && data[index]) {
+                                obj.updateCell(cell[0]+i, cell[1]+j, data[index]);
+                            }
+                            index++;
+                        }
+                    }
+                }
+    
+                // Update selection
+                obj.updateSelection(obj.records[cell[1]][cell[0]], obj.records[cell[1]+j-1][cell[0]+i-1]);
+    
+                if (! keepOptions) {
+                    delete(obj.options.mergeCells[cellName]);
+                }
+            }
+        }
+    
+        /**
+         * Remove all merged cells
+         */
+        obj.destroyMerged = function(keepOptions) {
+            // Remove any merged cells
+            if (obj.options.mergeCells) {
+                var mergedCells = obj.options.mergeCells;
+                var keys = Object.keys(obj.options.mergeCells);
+                for (var i = 0; i < keys.length; i++) {
+                    obj.removeMerge(keys[i], null, keepOptions);
+                }
+            }
+        }
+    
+        /**
+         * Is column merged
+         */
+        obj.isColMerged = function(x, insertBefore) {
+            var cols = [];
+            // Remove any merged cells
+            if (obj.options.mergeCells) {
+                var keys = Object.keys(obj.options.mergeCells);
+                for (var i = 0; i < keys.length; i++) {
+                    var info = jexcel.getIdFromColumnName(keys[i], true);
+                    var colspan = obj.options.mergeCells[keys[i]][0];
+                    var x1 = info[0];
+                    var x2 = info[0] + (colspan > 1 ? colspan - 1 : 0);
+    
+                    if (insertBefore == null) {
+                        if ((x1 <= x && x2 >= x)) {
+                            cols.push(keys[i]);
+                        }
+                    } else {
+                        if (insertBefore) {
+                            if ((x1 < x && x2 >= x)) {
+                                cols.push(keys[i]);
+                            }
+                        } else {
+                            if ((x1 <= x && x2 > x)) {
+                                cols.push(keys[i]);
+                            }
+                        }
+                    }
+                }
+            }
+    
+            return cols;
+        }
+    
+        /**
+         * Is rows merged
+         */
+        obj.isRowMerged = function(y, insertBefore) {
+            var rows = [];
+            // Remove any merged cells
+            if (obj.options.mergeCells) {
+                var keys = Object.keys(obj.options.mergeCells);
+                for (var i = 0; i < keys.length; i++) {
+                    var info = jexcel.getIdFromColumnName(keys[i], true);
+                    var rowspan = obj.options.mergeCells[keys[i]][1];
+                    var y1 = info[1];
+                    var y2 = info[1] + (rowspan > 1 ? rowspan - 1 : 0);
+    
+                    if (insertBefore == null) {
+                        if ((y1 <= y && y2 >= y)) {
+                            rows.push(keys[i]);
+                        }
+                    } else {
+                        if (insertBefore) {
+                            if ((y1 < y && y2 >= y)) {
+                                rows.push(keys[i]);
+                            }
+                        } else {
+                            if ((y1 <= y && y2 > y)) {
+                                rows.push(keys[i]);
+                            }
+                        }
+                    }
+                }
+            }
+    
+            return rows;
+        }
+
+        /**
+         * Open the column filter
+         */
+        obj.openFilter = function(columnId) {
+            if (! obj.options.filters) {
+                console.log('JEXCEL: filters not enabled.');
+            } else {
+                // Make sure is integer
+                columnId = parseInt(columnId);
+                // Reset selection
+                obj.resetSelection();
+                // Load options
+                var options = [];
+                for (var j = 0; j < obj.options.data.length; j++) {
+                    var k = obj.options.data[j][columnId];
+                    var v = obj.records[j][columnId].innerHTML;
+                    if (k && v) {
+                        options[k] = v;
+                    }
+                }
+                var keys = Object.keys(options);
+                var optionsFiltered = [];
+                for (var j = 0; j < keys.length; j++) {
+                    optionsFiltered.push({ id: keys[j], name: options[keys[j]] });
+                }
+
+                // Create dropdown
+                var div = document.createElement('div');
+                obj.filter.children[columnId + 1].innerHTML = '';
+                obj.filter.children[columnId + 1].appendChild(div);
+                obj.filter.children[columnId + 1].style.paddingLeft = '0px';
+                obj.filter.children[columnId + 1].style.paddingRight = '0px';
+                obj.filter.children[columnId + 1].style.overflow = 'initial';
+
+                // Dynamic dropdown
+                jSuites.dropdown(div, {
+                    data: optionsFiltered,
+                    multiple: true,
+                    autocomplete: true,
+                    opened: true,
+                    value: obj.filters[columnId] ? obj.filters[columnId] : '',
+                    width:'100%',
+                    position: (obj.options.tableOverflow == true || obj.options.fullscreen == true) ? true : false,
+                    onclose: function(o) {
+                        obj.resetFilters();
+                        obj.filters[columnId] = o.dropdown.getValue(true);
+                        obj.filter.children[columnId + 1].innerHTML = o.dropdown.getText();
+                        obj.filter.children[columnId + 1].style.paddingLeft = '';
+                        obj.filter.children[columnId + 1].style.paddingRight = '';
+                        obj.filter.children[columnId + 1].style.overflow = '';
+                        obj.closeFilter(columnId);
+                    }
+                });
+            }
+        }
+
+        obj.resetFilters = function() {
+            if (obj.options.filters) {
+                for (var i = 0; i < obj.filter.children.length; i++) {
+                    obj.filter.children[i].innerHTML = '&nbsp;';
+                    obj.filters[i] = null;
+                }
+            }
+        }
+
+        obj.closeFilter = function(columnId) {
+            if (! columnId) {
+                for (var i = 0; i < obj.filter.children.length; i++) {
+                    if (obj.filters[i]) {
+                        columnId = i;
+                    }
+                }
+            }
+
+            // Search filter
+            var search = function(query, x, y) {
+                for (var i = 0; i < query.length; i++) {
+                    if ((''+obj.options.data[y][x]).search(query[i]) >= 0 ||
+                        (''+obj.records[y][x].innerHTML).search(query[i]) >= 0) {
+                        return true;
+                    }
+                }
+                return false;
+            }
+
+            var query = obj.filters[columnId];
+            obj.results = [];
+            for (var j = 0; j < obj.options.data.length; j++) {
+                if (search(query, columnId, j)) {
+                    obj.results.push(j);
+                }
+            }
+            if (! obj.results.length) {
+                obj.results = null;
+            }
+
+            obj.updateResult();
+        }
+
+        /**
+         * Open the editor
+         * 
+         * @param object cell
+         * @return void
+         */
+        obj.openEditor = function(cell, empty, e) {
+            // Get cell position
+            var y = cell.getAttribute('data-y');
+            var x = cell.getAttribute('data-x');
+    
+            // On edition start
+            obj.dispatch('oneditionstart', el, cell, x, y);
+    
+            // Overflow
+            if (x > 0) {
+                obj.records[y][x-1].style.overflow = 'hidden';
+            }
+    
+            // Create editor
+            var createEditor = function(type) {
+                // Cell information
+                var info = cell.getBoundingClientRect();
+    
+                // Create dropdown
+                var editor = document.createElement(type);
+                editor.style.width = (info.width) + 'px';
+                editor.style.height = (info.height - 2) + 'px';
+                editor.style.minHeight = (info.height - 2) + 'px';
+    
+                // Edit cell
+                cell.classList.add('editor');
+                cell.innerHTML = '';
+                cell.appendChild(editor);
+    
+                return editor;
+            }
+    
+            // Readonly
+            if (cell.classList.contains('readonly') == true) {
+                // Do nothing
+            } else {
+                // Holder
+                obj.edition = [ obj.records[y][x], obj.records[y][x].innerHTML, x, y ];
+    
+                // If there is a custom editor for it
+                if (obj.options.columns[x].editor) {
+                    // Custom editors
+                    obj.options.columns[x].editor.openEditor(cell, el, empty, e);
+                } else {
+                    // Native functions
+                    if (obj.options.columns[x].type == 'hidden') {
+                        // Do nothing
+                    } else if (obj.options.columns[x].type == 'checkbox' || obj.options.columns[x].type == 'radio') {
+                        // Get value
+                        var value = cell.children[0].checked ? false : true;
+                        // Toogle value
+                        obj.setValue(cell, value);
+                        // Do not keep edition open
+                        obj.edition = null;
+                    } else if (obj.options.columns[x].type == 'dropdown' || obj.options.columns[x].type == 'autocomplete') {
+                        // Get current value
+                        var value = obj.options.data[y][x];
+    
+                        // Create dropdown
+                        if (typeof(obj.options.columns[x].filter) == 'function') {
+                            var source = obj.options.columns[x].filter(el, cell, x, y, obj.options.columns[x].source);
+                        } else {
+                            var source = obj.options.columns[x].source;
+                        }
+    
+                        // Create editor
+                        var editor = createEditor('div');
+                        var options = {
+                            data: source,
+                            multiple: obj.options.columns[x].multiple ? true : false,
+                            autocomplete: obj.options.columns[x].autocomplete || obj.options.columns[x].type == 'autocomplete' ? true : false,
+                            opened:true,
+                            value: obj.options.columns[x].multiple ? value.split(';') : value,
+                            width:'100%',
+                            height:editor.style.minHeight,
+                            position: (obj.options.tableOverflow == true || obj.options.fullscreen == true) ? true : false,
+                            onclose:function() {
+                                obj.closeEditor(cell, true);
+                            }
+                        };
+                        if (obj.options.columns[x].options && obj.options.columns[x].options.type) {
+                            options.type = obj.options.columns[x].options.type;
+                        }
+                        jSuites.dropdown(editor, options);
+                    } else if (obj.options.columns[x].type == 'calendar' || obj.options.columns[x].type == 'color') {
+                        // Value
+                        var value = obj.options.data[y][x];
+                        // Create editor
+                        var editor = createEditor('input');
+                        editor.value = value;
+    
+                        if (obj.options.tableOverflow == true || obj.options.fullscreen == true) {
+                            obj.options.columns[x].options.position = true;
+                        }
+                        obj.options.columns[x].options.value = obj.options.data[y][x];
+                        obj.options.columns[x].options.opened = true;
+                        obj.options.columns[x].options.onclose = function(el, value) {
+                            obj.closeEditor(cell, true);
+                        }
+                        // Current value
+                        if (obj.options.columns[x].type == 'color') {
+                            jSuites.color(editor, obj.options.columns[x].options);
+                        } else {
+                            jSuites.calendar(editor, obj.options.columns[x].options);
+                        }
+                        // Focus on editor
+                        editor.focus();
+                    } else if (obj.options.columns[x].type == 'html') {
+                        var value = obj.options.data[y][x];
+                        // Create editor
+                        var editor = createEditor('div');
+                        editor.style.position = 'relative';
+                        var div = document.createElement('div');
+                        div.classList.add('jexcel_richtext');
+                        editor.appendChild(div);
+                        jSuites.editor(div, {
+                            focus: true,
+                            value: value,
+                        });
+                        const rect = cell.getBoundingClientRect();
+                        const rectContent = div.getBoundingClientRect();
+                        if (window.innerHeight < rect.bottom + rectContent.height) {
+                            div.style.top = (rect.top - (rectContent.height + 2)) + 'px';
+                        } else {
+                            div.style.top = (rect.top) + 'px';
+                        }
+                    } else if (obj.options.columns[x].type == 'image') {
+                        // Value
+                        var img = cell.children[0];
+                        // Create editor
+                        var editor = createEditor('div');
+                        editor.style.position = 'relative';
+                        var div = document.createElement('div');
+                        div.classList.add('jclose');
+                        if (img && img.src) {
+                            div.appendChild(img);
+                        }
+                        editor.appendChild(div);
+                        jSuites.image(div, obj.options.imageOptions);
+                        const rect = cell.getBoundingClientRect();
+                        const rectContent = div.getBoundingClientRect();
+                        if (window.innerHeight < rect.bottom + rectContent.height) {
+                            div.style.top = (rect.top - (rectContent.height + 2)) + 'px';
+                        } else {
+                            div.style.top = (rect.top) + 'px';
+                        }
+                    } else {
+                        // Value
+                        var value = empty == true ? '' : obj.options.data[y][x];
+    
+                        // Basic editor
+                        if (obj.options.columns[x].wordWrap != false && (obj.options.wordWrap == true || obj.options.columns[x].wordWrap == true)) {
+                            var editor = createEditor('textarea');
+                        } else {
+                            var editor = createEditor('input');
+                            // Mask
+                            if (obj.options.columns[x].mask) {
+                                editor.setAttribute('data-mask', obj.options.columns[x].mask);
+                            }
+                        }
+
+                        editor.onblur = function() {
+                            obj.closeEditor(cell, true);
+                        };
+                        editor.focus();
+                        editor.value = value;
+                    }
+                }
+            }
+        }
+    
+        /**
+         * Close the editor and save the information
+         * 
+         * @param object cell
+         * @param boolean save
+         * @return void
+         */
+        obj.closeEditor = function(cell, save) {
+            var x = parseInt(cell.getAttribute('data-x'));
+            var y = parseInt(cell.getAttribute('data-y'));
+
+            // Get cell properties
+            if (save == true) {
+                // If custom editor
+                if (obj.options.columns[x].editor) {
+                    // Custom editor
+                    var value = obj.options.columns[x].editor.closeEditor(cell, save);
+                } else {
+                    // Native functions
+                    if (obj.options.columns[x].type == 'checkbox' || obj.options.columns[x].type == 'radio' || obj.options.columns[x].type == 'hidden') {
+                        // Do nothing
+                    } else if (obj.options.columns[x].type == 'dropdown' || obj.options.columns[x].type == 'autocomplete') {
+                        var value = cell.children[0].dropdown.close(true);
+                    } else if (obj.options.columns[x].type == 'calendar') {
+                        var value = cell.children[0].calendar.close(true);
+                    } else if (obj.options.columns[x].type == 'color') {
+                        var value = cell.children[0].color.close(true);
+                    } else if (obj.options.columns[x].type == 'html') {
+                        var value = cell.children[0].children[0].editor.getData();
+                    } else if (obj.options.columns[x].type == 'image') {
+                        var img = cell.children[0].children[0].children[0];
+                        var value = img && img.tagName == 'IMG' ? img.src : '';
+                    } else if (obj.options.columns[x].type == 'numeric') {
+                        var value = cell.children[0].value;
+                        if (value.substr(0,1) != '=') {
+                            if (value == '') {
+                                value = obj.options.columns[x].allowEmpty ? '' : 0;
+                            }
+                        }
+                        cell.children[0].onblur = null;
+                    } else {
+                        var value = cell.children[0].value;
+                        cell.children[0].onblur = null;
+                    }
+                }
+
+                // Ignore changes if the value is the same
+                if (obj.options.data[y][x] == value) {
+                    cell.innerHTML = obj.edition[1];
+                } else {
+                    obj.setValue(cell, value);
+                }
+            } else {
+                if (obj.options.columns[x].editor) {
+                    // Custom editor
+                    obj.options.columns[x].editor.closeEditor(cell, save);
+                } else {
+                    if (obj.options.columns[x].type == 'dropdown' || obj.options.columns[x].type == 'autocomplete') {
+                        cell.children[0].dropdown.close(true);
+                    } else if (obj.options.columns[x].type == 'calendar') {
+                        cell.children[0].calendar.close(true);
+                    } else if (obj.options.columns[x].type == 'color') {
+                        cell.children[0].color.close(true);
+                    } else {
+                        cell.children[0].onblur = null;
+                    }
+                }
+    
+                // Restore value
+                cell.innerHTML = obj.edition && obj.edition[1] ? obj.edition[1] : '';
+            }
+    
+            // On edition end
+            obj.dispatch('oneditionend', el, cell, x, y, value, save);
+
+            // Remove editor class
+            cell.classList.remove('editor');
+    
+            // Finish edition
+            obj.edition = null;
+        }
+    
+        /**
+         * Get the cell object
+         * 
+         * @param object cell
+         * @return string value
+         */
+        obj.getCell = function(cell) {
+            // Convert in case name is excel liked ex. A10, BB92
+            cell = jexcel.getIdFromColumnName(cell, true);
+            var x = cell[0];
+            var y = cell[1];
+    
+            return obj.records[y][x];
+        }
+    
+        /**
+         * Get the cell object from coords
+         * 
+         * @param object cell
+         * @return string value
+         */
+        obj.getCellFromCoords = function(x, y) {
+            return obj.records[y][x];
+        }
+    
+        /**
+         * Get label
+         * 
+         * @param object cell
+         * @return string value
+         */
+        obj.getLabel = function(cell) {
+            // Convert in case name is excel liked ex. A10, BB92
+            cell = jexcel.getIdFromColumnName(cell, true);
+            var x = cell[0];
+            var y = cell[1];
+    
+            return obj.records[y][x].innerHTML;
+        }
+    
+        /**
+         * Get labelfrom coords
+         * 
+         * @param object cell
+         * @return string value
+         */
+        obj.getLabelFromCoords = function(x, y) {
+            return obj.records[y][x].innerHTML;
+        }
+    
+        /**
+         * Get the value from a cell
+         * 
+         * @param object cell
+         * @return string value
+         */
+        obj.getValue = function(cell, processedValue) {
+            if (typeof(cell) == 'object') {
+                var x = cell.getAttribute('data-x');
+                var y = cell.getAttribute('data-y');
+            } else {
+                cell = jexcel.getIdFromColumnName(cell, true);
+                var x = cell[0];
+                var y = cell[1];
+            }
+    
+            var value = null;
+    
+            if (x != null && y != null) {
+                if (obj.records[y] && obj.records[y][x] && (processedValue || obj.options.copyCompatibility == true)) {
+                    value = obj.records[y][x].innerHTML;
+                } else {
+                    if (obj.options.data[y] && obj.options.data[y][x] != 'undefined') {
+                        value = obj.options.data[y][x];
+                    }
+                }
+            }
+    
+            return value;
+        }
+    
+        /**
+         * Get the value from a coords
+         * 
+         * @param int x
+         * @param int y
+         * @return string value
+         */
+        obj.getValueFromCoords = function(x, y, processedValue) {
+            var value = null;
+    
+            if (x != null && y != null) {
+                if ((obj.records[y] && obj.records[y][x]) && processedValue || obj.options.copyCompatibility == true) {
+                    value = obj.records[y][x].innerHTML;
+                } else {
+                    if (obj.options.data[y] && obj.options.data[y][x] != 'undefined') {
+                        value = obj.options.data[y][x];
+                    }
+                }
+            }
+    
+            return value;
+        }
+    
+        /**
+         * Set a cell value
+         * 
+         * @param mixed cell destination cell
+         * @param string value value
+         * @return void
+         */
+        obj.setValue = function(cell, value, force) {
+            var records = [];
+    
+            if (typeof(cell) == 'string') {
+                var columnId = jexcel.getIdFromColumnName(cell, true);
+                var x = columnId[0];
+                var y = columnId[1];
+    
+                // Update cell
+                records.push(obj.updateCell(x, y, value, force));
+    
+                // Update all formulas in the chain
+                obj.updateFormulaChain(x, y, records);
+            } else {
+                var x = null;
+                var y = null;
+                if (cell && cell.getAttribute) {
+                    var x = cell.getAttribute('data-x');
+                    var y = cell.getAttribute('data-y');
+                }
+    
+                // Update cell
+                if (x != null && y != null) {
+                    records.push(obj.updateCell(x, y, value, force));
+    
+                    // Update all formulas in the chain
+                    obj.updateFormulaChain(x, y, records);
+                } else {
+                    var keys = Object.keys(cell);
+                    if (keys.length > 0) {
+                        for (var i = 0; i < keys.length; i++) {
+                            if (typeof(cell[i]) == 'string') {
+                                var columnId = jexcel.getIdFromColumnName(cell[i], true);
+                                var x = columnId[0];
+                                var y = columnId[1];
+                            } else {
+                                if (cell[i].x != null && cell[i].y != null) {
+                                    var x = cell[i].x;
+                                    var y = cell[i].y;
+                                    // Flexible setup
+                                    if (cell[i].newValue != null) {
+                                        value = cell[i].newValue;
+                                    } else if (cell[i].value != null) {
+                                        value = cell[i].value;
+                                    }
+                                } else {
+                                    var x = cell[i].getAttribute('data-x');
+                                    var y = cell[i].getAttribute('data-y');
+                                }
+                            }
+    
+                             // Update cell
+                            if (x != null && y != null) {
+                                records.push(obj.updateCell(x, y, value, force));
+    
+                                // Update all formulas in the chain
+                                obj.updateFormulaChain(x, y, records);
+                            }
+                        }
+                    }
+                }
+            }
+    
+            // Update history
+            obj.setHistory({
+                action:'setValue',
+                records:records,
+                selection:obj.selectedCell,
+            });
+    
+            // Update table with custom configurations if applicable
+            obj.updateTable();
+    
+            // On after changes
+            obj.onafterchanges(el, records);
+        }
+    
+        /**
+         * Set a cell value based on coordinates
+         * 
+         * @param int x destination cell
+         * @param int y destination cell
+         * @param string value
+         * @return void
+         */
+        obj.setValueFromCoords = function(x, y, value, force) {
+            var records = [];
+            records.push(obj.updateCell(x, y, value, force));
+    
+            // Update all formulas in the chain
+            obj.updateFormulaChain(x, y, records);
+    
+            // Update history
+            obj.setHistory({
+                action:'setValue',
+                records:records,
+                selection:obj.selectedCell,
+            });
+    
+            // Update table with custom configurations if applicable
+            obj.updateTable();
+    
+            // On after changes
+            obj.onafterchanges(el, records);
+        }
+    
+        /**
+         * Toogle
+         */
+        obj.setCheckRadioValue = function() {
+            var records = [];
+            var keys = Object.keys(obj.highlighted);
+            for (var i = 0; i < keys.length; i++) {
+                var x = obj.highlighted[i].getAttribute('data-x');
+                var y = obj.highlighted[i].getAttribute('data-y');
+    
+                if (obj.options.columns[x].type == 'checkbox' || obj.options.columns[x].type == 'radio') {
+                    // Update cell
+                    records.push(obj.updateCell(x, y, ! obj.options.data[y][x]));
+                }
+            }
+    
+            if (records.length) {
+                // Update history
+                obj.setHistory({
+                    action:'setValue',
+                    records:records,
+                    selection:obj.selectedCell,
+                });
+    
+                // On after changes
+                obj.onafterchanges(el, records);
+            }
+        }
+    
+        /**
+         * Strip tags
+         */
+        var stripScript = function(a) {
+            var b = new Option;
+            b.innerHTML = a;
+            var c = null;
+            for (a = b.getElementsByTagName('script'); c=a[0];) c.parentNode.removeChild(c);
+            return b.innerHTML;
+        }
+
+        /**
+         * Update cell content
+         * 
+         * @param object cell
+         * @return void
+         */
+        obj.updateCell = function(x, y, value, force) {
+            // Changing value depending on the column type
+            if (obj.records[y][x].classList.contains('readonly') == true && ! force) {
+                // Do nothing
+                var record = {
+                    x: x,
+                    y: y,
+                    col: x,
+                    row: y
+                }
+            } else {
+                // Security
+                if ((''+value).substr(0,1) == '=' && obj.options.secureFormulas == true) {
+                    var val = secureFormula(value);
+                    if (val != value) {
+                        // Update the data container
+                        value = val;
+                    }
+                }
+
+                // On change
+                var val = obj.dispatch('onbeforechange', el, obj.records[y][x], x, y, value);
+
+                // If you return something this will overwrite the value
+                if (val != undefined) {
+                    value = val;
+                }
+
+                if (obj.options.columns[x].editor && typeof(obj.options.columns[x].editor.updateCell) == 'function') {
+                    value = obj.options.columns[x].editor.updateCell(obj.records[y][x], value, force);
+                }
+
+                // History format
+                var record = {
+                    x: x,
+                    y: y,
+                    col: x,
+                    row: y,
+                    newValue: value,
+                    oldValue: obj.options.data[y][x],
+                }
+
+                if (obj.options.columns[x].editor) {
+                    // Update data and cell
+                    obj.options.data[y][x] = value;
+                } else {
+                    // Native functions
+                    if (obj.options.columns[x].type == 'checkbox' || obj.options.columns[x].type == 'radio') {
+                        // Unchecked all options
+                        if (obj.options.columns[x].type == 'radio') {
+                            for (var j = 0; j < obj.options.data.length; j++) {
+                                obj.options.data[j][x] = false;
+                            }
+                        }
+
+                        // Update data and cell
+                        obj.records[y][x].children[0].checked = (value == 1 || value == true || value == 'true' || value == 'TRUE') ? true : false;
+                        obj.options.data[y][x] = obj.records[y][x].children[0].checked;
+                    } else if (obj.options.columns[x].type == 'dropdown' || obj.options.columns[x].type == 'autocomplete') {
+                        // Update data and cell
+                        obj.options.data[y][x] = value;
+                        obj.records[y][x].innerText = obj.getDropDownValue(x, value);
+                    } else if (obj.options.columns[x].type == 'calendar') {
+                        // Update calendar
+                        var formatted = jSuites.calendar.extractDateFromString(value, obj.options.columns[x].options.format);
+                        // Update data and cell
+                        obj.options.data[y][x] = value;
+                        obj.records[y][x].innerText = jSuites.calendar.getDateString(formatted ? formatted : value, obj.options.columns[x].options.format);
+                    } else if (obj.options.columns[x].type == 'color') {
+                        // Update color
+                        obj.options.data[y][x] = value;
+                        // Render
+                        if (obj.options.columns[x].render == 'square') {
+                            var color = document.createElement('div');
+                            color.className = 'color';
+                            color.style.backgroundColor = value;
+                            obj.records[y][x].innerText = '';
+                            obj.records[y][x].appendChild(color);
+                        } else {
+                        obj.records[y][x].style.color = value;
+                            obj.records[y][x].innerText = value;
+                        }
+                    } else if (obj.options.columns[x].type == 'image') {
+                        value = ''+value;
+                        obj.options.data[y][x] = value;
+                        obj.records[y][x].innerHTML = '';
+                        if (value && value.substr(0, 10) == 'data:image') {
+                            var img = document.createElement('img');
+                            img.src = value;
+                            obj.records[y][x].appendChild(img);
+                        }
+                    } else {
+                        // Update data and cell
+                        obj.options.data[y][x] = value;
+                        // Label
+                        if (obj.options.columns[x].type == 'html') {
+                            obj.records[y][x].innerHTML = stripScript(obj.parseValue(x, y, value));
+                        } else {
+                            if (obj.options.stripHTML === false || obj.options.columns[x].stripHTML === false) {
+                                obj.records[y][x].innerHTML = stripScript(obj.parseValue(x, y, value));
+                            } else {
+                                obj.records[y][x].innerText = obj.parseValue(x, y, value);
+                            }
+                        }
+                        // Handle big text inside a cell
+                        if (obj.options.columns[x].wordWrap != false && (obj.options.wordWrap == true || obj.options.columns[x].wordWrap == true || obj.records[y][x].innerHTML.length > 200)) {
+                            obj.records[y][x].style.whiteSpace = 'pre-wrap';
+                        } else {
+                            obj.records[y][x].style.whiteSpace = '';
+                        }
+                    }
+                }
+
+                // Overflow
+                if (x > 0) {
+                    if (value) {
+                        obj.records[y][x-1].style.overflow = 'hidden';
+                    } else {
+                        obj.records[y][x-1].style.overflow = '';
+                    }
+                }
+
+                // On change
+                obj.dispatch('onchange', el, (obj.records[y] && obj.records[y][x] ? obj.records[y][x] : null), x, y, value, record.oldValue);
+            }
+
+            return record;
+        }
+
+        /**
+         * Helper function to copy data using the corner icon
+         */
+        obj.copyData = function(o, d) {
+            // Get data from all selected cells
+            var data = obj.getData(true, true);
+
+            // Selected cells
+            var h = obj.selectedContainer;
+
+            // Cells
+            var x1 = parseInt(o.getAttribute('data-x'));
+            var y1 = parseInt(o.getAttribute('data-y'));
+            var x2 = parseInt(d.getAttribute('data-x'));
+            var y2 = parseInt(d.getAttribute('data-y'));
+
+            // Records
+            var records = [];
+            var breakControl = false;
+
+            if (h[0] == x1) {
+                // Vertical copy
+                if (y1 < h[1]) {
+                    var rowNumber = y1 - h[1];
+                } else {
+                    var rowNumber = 1;
+                }
+                var colNumber = 0;
+            } else {
+                if (x1 < h[0]) {
+                    var colNumber = x1 - h[0];
+                } else {
+                    var colNumber = 1;
+                }
+                var rowNumber = 0;
+            }
+
+            // Copy data procedure
+            var posx = 0;
+            var posy = 0;
+
+            for (var j = y1; j <= y2; j++) {
+                // Skip hidden rows
+                if (obj.rows[j] && obj.rows[j].style.display == 'none') {
+                    continue;
+                }
+
+                // Controls
+                if (data[posy] == undefined) {
+                    posy = 0;
+                }
+                posx = 0;
+
+                // Data columns
+                if (h[0] != x1) {
+                    if (x1 < h[0]) {
+                        var colNumber = x1 - h[0];
+                    } else {
+                        var colNumber = 1;
+                    }
+                }
+                // Data columns
+                for (var i = x1; i <= x2; i++) {
+                    // Update non-readonly
+                    if (obj.records[j][i] && ! obj.records[j][i].classList.contains('readonly') && obj.records[j][i].style.display != 'none' && breakControl == false) {
+                        // Stop if contains value
+                        if (! obj.selection.length) {
+                            if (obj.options.data[j][i] != '') {
+                                breakControl = true;
+                                continue;
+                            }
+                        }
+    
+                        // Column
+                        if (data[posy] == undefined) {
+                            posx = 0;
+                        } else if (data[posy][posx] == undefined) {
+                            posx = 0;
+                        }
+
+                        // Value
+                        var value = data[posy][posx];
+    
+                        if (value && ! data[1] && obj.options.autoIncrement == true) {
+                            if (obj.options.columns[i].type == 'text' || obj.options.columns[i].type == 'number') {
+                                if ((''+value).substr(0,1) == '=') {
+                                    var tokens = value.match(/([A-Z]+[0-9]+)/g);
+    
+                                    if (tokens) {
+                                        var affectedTokens = [];
+                                        for (var index = 0; index < tokens.length; index++) {
+                                            var position = jexcel.getIdFromColumnName(tokens[index], 1);
+                                            position[0] += colNumber;
+                                            position[1] += rowNumber;
+                                            if (position[1] < 0) {
+                                                position[1] = 0;
+                                            }
+                                            var token = jexcel.getColumnNameFromId([position[0], position[1]]);
+    
+                                            if (token != tokens[index]) {
+                                                affectedTokens[tokens[index]] = token;
+                                            }
+                                        }
+                                        // Update formula
+                                        if (affectedTokens) {
+                                            value = obj.updateFormula(value, affectedTokens)
+                                        }
+                                    }
+                                } else {
+                                    if (value == Number(value)) {
+                                        value = Number(value) + rowNumber;
+                                    }
+                                }
+                            } else if (obj.options.columns[i].type == 'calendar') {
+                                var date = new Date(value);
+                                date.setDate(date.getDate() + rowNumber);
+                                value = date.getFullYear() + '-' + jexcel.doubleDigitFormat(parseInt(date.getMonth() + 1)) + '-' + jexcel.doubleDigitFormat(date.getDate()) + ' ' + '00:00:00';
+                            }
+                        }
+    
+                        records.push(obj.updateCell(i, j, value));
+    
+                        // Update all formulas in the chain
+                        obj.updateFormulaChain(i, j, records);
+                    }
+                    posx++;
+                    if (h[0] != x1) {
+                        colNumber++;
+                    }
+                }
+                posy++;
+                rowNumber++;
+            }
+    
+            // Update history
+            obj.setHistory({
+                action:'setValue',
+                records:records,
+                selection:obj.selectedCell,
+            });
+    
+            // Update table with custom configuration if applicable
+            obj.updateTable();
+    
+            // On after changes
+            obj.onafterchanges(el, records);
+        }
+    
+        /**
+         * Refresh current selection
+         */
+        obj.refreshSelection = function() {
+            if (obj.selectedCell) {
+                obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+            }
+        }
+    
+        /**
+         * Move coords to A1 in case ovelaps with an excluded cell
+         */
+        obj.conditionalSelectionUpdate = function(type, o, d) {
+            if (type == 1) {
+                if (obj.selectedCell && ((o >= obj.selectedCell[1] && o <= obj.selectedCell[3]) || (d >= obj.selectedCell[1] && d <= obj.selectedCell[3]))) {
+                    obj.resetSelection();
+                    return;
+                }
+            } else {
+                if (obj.selectedCell && ((o >= obj.selectedCell[0] && o <= obj.selectedCell[2]) || (d >= obj.selectedCell[0] && d <= obj.selectedCell[2]))) {
+                    obj.resetSelection();
+                    return;
+                }
+            }
+        }
+    
+        /**
+         * Clear table selection
+         */
+        obj.resetSelection = function(blur) {
+            // Remove style
+            if (! obj.highlighted.length) {
+                var previousStatus = 0;
+            } else {
+                var previousStatus = 1;
+    
+                for (var i = 0; i < obj.highlighted.length; i++) {
+                    obj.highlighted[i].classList.remove('highlight');
+                    obj.highlighted[i].classList.remove('highlight-left');
+                    obj.highlighted[i].classList.remove('highlight-right');
+                    obj.highlighted[i].classList.remove('highlight-top');
+                    obj.highlighted[i].classList.remove('highlight-bottom');
+                    obj.highlighted[i].classList.remove('highlight-selected');
+    
+                    var px = parseInt(obj.highlighted[i].getAttribute('data-x'));
+                    var py = parseInt(obj.highlighted[i].getAttribute('data-y'));
+    
+                    // Check for merged cells
+                    if (obj.highlighted[i].getAttribute('data-merged')) {
+                        var colspan = parseInt(obj.highlighted[i].getAttribute('colspan'));
+                        var rowspan = parseInt(obj.highlighted[i].getAttribute('rowspan'));
+                        var ux = colspan > 0 ? px + (colspan - 1) : px;
+                        var uy = rowspan > 0 ? py + (rowspan - 1): py;
+                    } else {
+                        var ux = px;
+                        var uy = py;
+                    }
+    
+                    // Remove selected from headers
+                    for (var j = px; j <= ux; j++) {
+                        if (obj.headers[j]) {
+                            obj.headers[j].classList.remove('selected');
+                        }
+                    }
+    
+                    // Remove selected from rows
+                    for (var j = py; j <= uy; j++) {
+                        if (obj.rows[j]) {
+                            obj.rows[j].classList.remove('selected');
+                        }
+                    }
+                }
+            }
+    
+            // Reset highlighed cells
+            obj.highlighted = [];
+    
+            // Reset
+            obj.selectedCell = null;
+    
+            // Hide corner
+            obj.corner.style.top = '-2000px';
+            obj.corner.style.left = '-2000px';
+    
+            if (blur == true && previousStatus == 1) {
+                obj.dispatch('onblur', el);
+            }
+    
+            return previousStatus;
+        }
+    
+        /**
+         * Update selection based on two cells
+         */
+        obj.updateSelection = function(el1, el2, origin) {
+            var x1 = el1.getAttribute('data-x');
+            var y1 = el1.getAttribute('data-y');
+            if (el2) {
+                var x2 = el2.getAttribute('data-x');
+                var y2 = el2.getAttribute('data-y');
+            } else {
+                var x2 = x1;
+                var y2 = y1;
+            }
+    
+            obj.updateSelectionFromCoords(x1, y1, x2, y2, origin);
+        }
+    
+        /**
+         * Update selection from coords
+         */
+        obj.updateSelectionFromCoords = function(x1, y1, x2, y2, origin) {
+            // Reset Selection
+            var updated = null;
+            var previousState = obj.resetSelection();
+    
+            // Same element
+            if (x2 == null) {
+                x2 = x1;
+            }
+            if (y2 == null) {
+                y2 = y1;
+            }
+    
+            // Selection must be within the existing data
+            if (x1 >= obj.headers.length) {
+                x1 = obj.headers.length - 1;
+            }
+            if (y1 >= obj.rows.length) {
+                y1 = obj.rows.length - 1;
+            }
+            if (x2 >= obj.headers.length) {
+                x2 = obj.headers.length - 1;
+            }
+            if (y2 >= obj.rows.length) {
+                y2 = obj.rows.length - 1;
+            }
+    
+            // Keep selected cell
+            obj.selectedCell = [x1, y1, x2, y2];
+    
+            // Select cells
+            if (x1 != null) {
+                // Add selected cell
+                if (obj.records[y1][x1]) {
+                    obj.records[y1][x1].classList.add('highlight-selected');
+                }
+    
+                // Origin & Destination
+                if (parseInt(x1) < parseInt(x2)) {
+                    var px = parseInt(x1);
+                    var ux = parseInt(x2);
+                } else {
+                    var px = parseInt(x2);
+                    var ux = parseInt(x1);
+                }
+    
+                if (parseInt(y1) < parseInt(y2)) {
+                    var py = parseInt(y1);
+                    var uy = parseInt(y2);
+                } else {
+                    var py = parseInt(y2);
+                    var uy = parseInt(y1);
+                }
+    
+                // Verify merged columns
+                for (var i = px; i <= ux; i++) {
+                    for (var j = py; j <= uy; j++) {
+                        if (obj.records[j][i] && obj.records[j][i].getAttribute('data-merged')) {
+                            var x = parseInt(obj.records[j][i].getAttribute('data-x'));
+                            var y = parseInt(obj.records[j][i].getAttribute('data-y'));
+                            var colspan = parseInt(obj.records[j][i].getAttribute('colspan'));
+                            var rowspan = parseInt(obj.records[j][i].getAttribute('rowspan'));
+    
+                            if (colspan > 1) {
+                                if (x < px) {
+                                    px = x;
+                                }
+                                if (x + colspan > ux) {
+                                    ux = x + colspan - 1;
+                                }
+                            }
+    
+                            if (rowspan) {
+                                if (y < py) {
+                                    py = y;
+    
+                                }
+                                if (y + rowspan > uy) {
+                                    uy = y + rowspan - 1;
+                                }
+                            }
+                        }
+                    }
+                }
+    
+                // Limits
+                var borderLeft = null;
+                var borderRight = null;
+                var borderTop = null;
+                var borderBottom = null;
+    
+                // Vertical limits
+                for (var j = py; j <= uy; j++) {
+                    if (obj.rows[j].style.display != 'none') {
+                        if (borderTop == null) {
+                            borderTop = j;
+                        }
+                        borderBottom = j;
+                    }
+                }
+    
+                // Redefining styles
+                for (var i = px; i <= ux; i++) {
+                    for (var j = py; j <= uy; j++) {
+                        if (obj.rows[j].style.display != 'none' && obj.records[j][i].style.display != 'none') {
+                            obj.records[j][i].classList.add('highlight');
+                            obj.highlighted.push(obj.records[j][i]);
+                        }
+                    }
+    
+                    // Horizontal limits
+                    if (obj.options.columns[i].type != 'hidden') {
+                        if (borderLeft == null) {
+                            borderLeft = i;
+                        }
+                        borderRight = i;
+                    }
+                }
+    
+                // Create borders
+                if (! borderLeft) {
+                    borderLeft = 0;
+                }
+                if (! borderRight) {
+                    borderRight = 0;
+                }
+                for (var i = borderLeft; i <= borderRight; i++) {
+                    if (obj.options.columns[i].type != 'hidden') {
+                        // Top border
+                        if (obj.records[borderTop][i]) {
+                            obj.records[borderTop][i].classList.add('highlight-top');
+                        }
+                        // Bottom border
+                        if (obj.records[borderBottom][i]) {
+                            obj.records[borderBottom][i].classList.add('highlight-bottom');
+                        }
+                        // Add selected from headers
+                        obj.headers[i].classList.add('selected');
+                    }
+                }
+    
+                for (var j = borderTop; j <= borderBottom; j++) {
+                    if (obj.rows[j].style.display != 'none') {
+                        // Left border
+                        obj.records[j][borderLeft].classList.add('highlight-left');
+                        // Right border
+                        obj.records[j][borderRight].classList.add('highlight-right');
+                        // Add selected from rows
+                        obj.rows[j].classList.add('selected');
+                    }
+                }
+    
+                obj.selectedContainer = [ borderLeft, borderTop, borderRight, borderBottom ];
+            }
+    
+            // Handle events
+            if (previousState == 0) {
+                obj.dispatch('onfocus', el);
+
+                obj.removeCopyingSelection();
+            }
+
+            obj.dispatch('onselection', el, borderLeft, borderTop, borderRight, borderBottom, origin);
+
+            // Find corner cell
+            obj.updateCornerPosition();
+        }
+    
+        /**
+         * Remove copy selection
+         * 
+         * @return void
+         */
+        obj.removeCopySelection = function() {
+            // Remove current selection
+            for (var i = 0; i < obj.selection.length; i++) {
+                obj.selection[i].classList.remove('selection');
+                obj.selection[i].classList.remove('selection-left');
+                obj.selection[i].classList.remove('selection-right');
+                obj.selection[i].classList.remove('selection-top');
+                obj.selection[i].classList.remove('selection-bottom');
+            }
+    
+            obj.selection = [];
+        }
+    
+        /**
+         * Update copy selection
+         * 
+         * @param int x, y
+         * @return void
+         */
+        obj.updateCopySelection = function(x3, y3) {
+            // Remove selection
+            obj.removeCopySelection();
+    
+            // Get elements first and last
+            var x1 = obj.selectedContainer[0];
+            var y1 = obj.selectedContainer[1];
+            var x2 = obj.selectedContainer[2];
+            var y2 = obj.selectedContainer[3];
+    
+            if (x3 != null && y3 != null) {
+                if (x3 - x2 > 0) {
+                    var px = parseInt(x2) + 1;
+                    var ux = parseInt(x3);
+                } else {
+                    var px = parseInt(x3);
+                    var ux = parseInt(x1) - 1;
+                }
+    
+                if (y3 - y2 > 0) {
+                    var py = parseInt(y2) + 1;
+                    var uy = parseInt(y3);
+                } else {
+                    var py = parseInt(y3);
+                    var uy = parseInt(y1) - 1;
+                }
+    
+                if (ux - px <= uy - py) {
+                    var px = parseInt(x1);
+                    var ux = parseInt(x2);
+                } else {
+                    var py = parseInt(y1);
+                    var uy = parseInt(y2);
+                }
+    
+                for (var j = py; j <= uy; j++) {
+                    for (var i = px; i <= ux; i++) {
+                        if (obj.records[j][i] && obj.rows[j].style.display != 'none' && obj.records[j][i].style.display != 'none') {
+                            obj.records[j][i].classList.add('selection');
+                            obj.records[py][i].classList.add('selection-top');
+                            obj.records[uy][i].classList.add('selection-bottom');
+                            obj.records[j][px].classList.add('selection-left');
+                            obj.records[j][ux].classList.add('selection-right');
+    
+                            // Persist selected elements
+                            obj.selection.push(obj.records[j][i]);
+                        }
+                    }
+                }
+            }
+        }
+    
+        /**
+         * Update corner position
+         * 
+         * @return void
+         */
+        obj.updateCornerPosition = function() {
+            // If any selected cells
+            if (! obj.highlighted.length) {
+                obj.corner.style.top = '-2000px';
+                obj.corner.style.left = '-2000px';
+            } else {
+                // Get last cell
+                var last = obj.highlighted[obj.highlighted.length-1];
+
+                const contentRect = obj.content.getBoundingClientRect();
+                var x1 = contentRect.left;
+                var y1 = contentRect.top;
+
+                const lastRect = last.getBoundingClientRect();
+                var x2 = lastRect.left;
+                var y2 = lastRect.top;
+                var w2 = lastRect.width;
+                var h2 = lastRect.height;
+
+                var x = (x2 - x1) + obj.content.scrollLeft + w2 - 4;
+                var y = (y2 - y1) + obj.content.scrollTop + h2 - 4;
+
+                // Place the corner in the correct place
+                obj.corner.style.top = y + 'px';
+                obj.corner.style.left = x + 'px';
+
+                if (obj.options.freezeColumns) {
+                    var width = obj.getFreezeWidth();
+                    if (x2 - x1 + w2 < width) {
+                        obj.corner.style.display = 'none';
+                    } else {
+                        obj.corner.style.display = '';
+                    }
+                } else {
+                    obj.corner.style.display = '';
+                }
+            }
+        }
+    
+        /**
+         * Update scroll position based on the selection
+         */
+        obj.updateScroll = function(direction) {
+            // jExcel Container information
+            const contentRect = obj.content.getBoundingClientRect();
+            var x1 = contentRect.left;
+            var y1 = contentRect.top;
+            var w1 = contentRect.width;
+            var h1 = contentRect.height;
+
+            // Direction Left or Up
+            var reference = obj.records[obj.selectedCell[3]][obj.selectedCell[2]];
+    
+            // Reference
+            const referenceRect = reference.getBoundingClientRect();
+            var x2 = referenceRect.left;
+            var y2 = referenceRect.top;
+            var w2 = referenceRect.width;
+            var h2 = referenceRect.height;
+    
+            // Direction
+            if (direction == 0 || direction == 1) {
+                var x = (x2 - x1) + obj.content.scrollLeft;
+                var y = (y2 - y1) + obj.content.scrollTop - 2;
+            } else {
+                var x = (x2 - x1) + obj.content.scrollLeft + w2;
+                var y = (y2 - y1) + obj.content.scrollTop + h2;
+            }
+    
+            // Top position check
+            if (y > (obj.content.scrollTop + 30) && y < (obj.content.scrollTop + h1)) {
+                // In the viewport
+            } else {
+                // Out of viewport
+                if (y < obj.content.scrollTop + 30) {
+                    obj.content.scrollTop = y - h2;
+                } else {
+                    obj.content.scrollTop = y - (h1 - 2);
+                }
+            }
+    
+            // Freeze columns? 
+            var freezed = obj.getFreezeWidth();
+
+            // Left position check - TODO: change that to the bottom border of the element
+            if (x > (obj.content.scrollLeft + freezed) && x < (obj.content.scrollLeft + w1)) {
+                // In the viewport
+            } else {
+                // Out of viewport
+                if (x < obj.content.scrollLeft + 30) {
+                    obj.content.scrollLeft = x;
+                    if (obj.content.scrollLeft < 50) {
+                        obj.content.scrollLeft = 0;
+                    }
+                } else if (x < obj.content.scrollLeft + freezed) {
+                    obj.content.scrollLeft = x - freezed - 1;
+                } else {
+                    obj.content.scrollLeft = x - (w1 - 20);
+                }
+            }
+        }
+    
+        /**
+         * Get the column width
+         * 
+         * @param int column column number (first column is: 0)
+         * @return int current width
+         */
+        obj.getWidth = function(column) {
+            if (! column) {
+                // Get all headers
+                var data = [];
+                for (var i = 0; i < obj.headers.length; i++) {
+                    data.push(obj.options.columns[i].width);
+                }
+            } else {
+                // In case the column is an object
+                if (typeof(column) == 'object') {
+                    column = $(column).getAttribute('data-x');
+                }
+    
+                data = obj.colgroup[column].getAttribute('width')
+            }
+    
+            return data;
+        }
+
+
+        /**
+         * Set the column width
+         * 
+         * @param int column number (first column is: 0)
+         * @param int new column width
+         * @param int old column width
+         */
+        obj.setWidth = function (column, width, oldWidth) {
+            if (width) {
+                if (Array.isArray(column)) {
+                    // Oldwidth
+                    if (! oldWidth) {
+                        var oldWidth = [];
+                    }
+                    // Set width
+                    for (var i = 0; i < column.length; i++) {
+                        if (! oldWidth[i]) {
+                            oldWidth[i] = obj.colgroup[column[i]].getAttribute('width');
+                        }
+                        var w = Array.isArray(width) && width[i] ? width[i] : width;
+                        obj.colgroup[column[i]].setAttribute('width', w);
+                        obj.options.columns[column[i]].width = w;
+                    }
+                } else {
+                    // Oldwidth
+                    if (! oldWidth) {
+                        oldWidth = obj.colgroup[column].getAttribute('width');
+                    }
+                    // Set width
+                    obj.colgroup[column].setAttribute('width', width);
+                    obj.options.columns[column].width = width;
+                }
+
+                // Keeping history of changes
+                obj.setHistory({
+                    action:'setWidth',
+                    column:column,
+                    oldValue:oldWidth,
+                    newValue:width,
+                });
+
+                // On resize column
+                obj.dispatch('onresizecolumn', el, column, width, oldWidth);
+
+                // Update corner position
+                obj.updateCornerPosition();
+            }
+        }
+
+        /**
+         * Set the row height
+         * 
+         * @param row - row number (first row is: 0)
+         * @param height - new row height
+         * @param oldHeight - old row height
+         */
+        obj.setHeight = function (row, height, oldHeight) {
+            if (height > 0) {
+                // In case the column is an object
+                if (typeof(row) == 'object') {
+                    row = row.getAttribute('data-y');
+                }
+    
+                // Oldwidth
+                if (! oldHeight) {
+                    oldHeight = obj.rows[row].getAttribute('height');
+
+                    if (! oldHeight) {
+                        var rect = obj.rows[row].getBoundingClientRect();
+                        oldHeight = rect.height;
+                    }
+                }
+
+                // Integer
+                height = parseInt(height);
+
+                // Set width
+                obj.rows[row].style.height = height + 'px';
+    
+                // Keep options updated
+                if (! obj.options.rows[row]) {
+                    obj.options.rows[row] = {};
+                }
+                obj.options.rows[row].height = height + 'px';
+    
+                // Keeping history of changes
+                obj.setHistory({
+                    action:'setHeight',
+                    row:row,
+                    oldValue:oldHeight,
+                    newValue:height,
+                });
+
+                // On resize column
+                obj.dispatch('onresizerow', el, row, height, oldHeight);
+
+                // Update corner position
+                obj.updateCornerPosition();
+            }
+        }
+    
+        /**
+         * Get the row height
+         * 
+         * @param row - row number (first row is: 0)
+         * @return height - current row height
+         */
+        obj.getHeight = function(row) {
+            if (! row) {
+                // Get height of all rows
+                var data = [];
+                for (var j = 0; j < obj.rows.length; j++) {
+                    var h = obj.rows[j].style.height;
+                    if (h) {
+                        data[j] = h;
+                    }
+                }
+            } else {
+                // In case the row is an object
+                if (typeof(row) == 'object') {
+                    row = $(row).getAttribute('data-y');
+                }
+    
+                var data = obj.rows[row].style.height;
+            }
+    
+            return data;
+        }
+    
+        obj.setFooter = function(data) {
+            if (data) {
+                obj.options.footers = data;
+            }
+
+            if (obj.options.footers) {
+                if (! obj.tfoot) {
+                    obj.tfoot = document.createElement('tfoot');
+                    obj.table.appendChild(obj.tfoot);
+                } 
+
+                for (var j = 0; j < obj.options.footers.length; j++) {
+                    if (obj.tfoot.children[j]) {
+                        var tr = obj.tfoot.children[j];
+                    } else {
+                        var tr = document.createElement('tr');
+                        var td = document.createElement('td');
+                        tr.appendChild(td);
+                        obj.tfoot.appendChild(tr);
+                    }
+                    for (var i = 0; i < obj.headers.length; i++) {
+                        if (! obj.options.footers[j][i]) {
+                            obj.options.footers[j][i] = '';
+                        }
+                        if (obj.tfoot.children[j].children[i+1]) {
+                            var td = obj.tfoot.children[j].children[i+1];
+                        } else {
+                            var td = document.createElement('td');
+                            tr.appendChild(td);
+
+                            // Text align
+                            var colAlign = obj.options.columns[i].align ? obj.options.columns[i].align : 'center';
+                            td.style.textAlign = colAlign;
+                        }
+                        td.innerText = obj.parseValue(i, j, obj.options.footers[j][i]);
+                    }
+                }
+            }
+        }
+
+        /**
+         * Get the column title
+         * 
+         * @param column - column number (first column is: 0)
+         * @param title - new column title
+         */
+        obj.getHeader = function(column) {
+            return obj.headers[column].innerText;
+        }
+    
+        /**
+         * Set the column title
+         * 
+         * @param column - column number (first column is: 0)
+         * @param title - new column title
+         */
+        obj.setHeader = function(column, newValue) {
+            if (obj.headers[column]) {
+                var oldValue = obj.headers[column].innerText;
+    
+                if (! newValue) {
+                    newValue = prompt(obj.options.text.columnName, oldValue)
+                }
+    
+                if (newValue) {
+                    obj.headers[column].innerText = newValue;
+                    // Keep the title property
+                    obj.headers[column].setAttribute('title', newValue);
+                    // Update title
+                    obj.options.columns[column].title = newValue;
+                }
+    
+                obj.setHistory({
+                    action: 'setHeader',
+                    column: column,
+                    oldValue: oldValue,
+                    newValue: newValue
+                });
+    
+                // On onchange header
+                obj.dispatch('onchangeheader', el, column, oldValue, newValue);
+            }
+        }
+    
+        /**
+         * Get the headers
+         * 
+         * @param asArray
+         * @return mixed
+         */
+        obj.getHeaders = function (asArray) {
+            var title = [];
+    
+            for (var i = 0; i < obj.headers.length; i++) {
+                title.push(obj.getHeader(i));
+            }
+    
+            return asArray ? title : title.join(obj.options.csvDelimiter);
+        }
+    
+        /**
+         * Get meta information from cell(s)
+         * 
+         * @return integer
+         */
+        obj.getMeta = function(cell, key) {
+            if (! cell) {
+                return obj.options.meta;
+            } else {
+                if (key) {
+                    return obj.options.meta[cell] && obj.options.meta[cell][key] ? obj.options.meta[cell][key] : null;
+                } else {
+                    return obj.options.meta[cell] ? obj.options.meta[cell] : null;
+                }
+            }
+        }
+    
+        /**
+         * Set meta information to cell(s)
+         * 
+         * @return integer
+         */
+        obj.setMeta = function(o, k, v) {
+            if (! obj.options.meta) {
+                obj.options.meta = {}
+            }
+    
+            if (k && v) {
+                // Set data value
+                if (! obj.options.meta[o]) {
+                    obj.options.meta[o] = {};
+                }
+                obj.options.meta[o][k] = v;
+            } else {
+                // Apply that for all cells
+                var keys = Object.keys(o);
+                for (var i = 0; i < keys.length; i++) {
+                    if (! obj.options.meta[keys[i]]) {
+                        obj.options.meta[keys[i]] = {};
+                    }
+    
+                    var prop = Object.keys(o[keys[i]]);
+                    for (var j = 0; j < prop.length; j++) {
+                        obj.options.meta[keys[i]][prop[j]] = o[keys[i]][prop[j]];
+                    }
+                }
+            }
+    
+            obj.dispatch('onchangemeta', el, o, k, v);
+        }
+    
+        /**
+         * Update meta information
+         * 
+         * @return integer
+         */
+        obj.updateMeta = function(affectedCells) {
+            if (obj.options.meta) {
+                var newMeta = {};
+                var keys = Object.keys(obj.options.meta);
+                for (var i = 0; i < keys.length; i++) {
+                    if (affectedCells[keys[i]]) {
+                        newMeta[affectedCells[keys[i]]] = obj.options.meta[keys[i]];
+                    } else {
+                        newMeta[keys[i]] = obj.options.meta[keys[i]];
+                    }
+                }
+                // Update meta information
+                obj.options.meta = newMeta;
+            }
+        }
+    
+        /**
+         * Get style information from cell(s)
+         * 
+         * @return integer
+         */
+        obj.getStyle = function(cell, key) {
+            // Cell
+            if (! cell) {
+                // Control vars
+                var data = {};
+    
+                // Column and row length
+                var x = obj.options.data[0].length;
+                var y = obj.options.data.length;
+    
+                // Go through the columns to get the data
+                for (var j = 0; j < y; j++) {
+                    for (var i = 0; i < x; i++) {
+                        // Value
+                        var v = key ? obj.records[j][i].style[key] : obj.records[j][i].getAttribute('style');
+    
+                        // Any meta data for this column?
+                        if (v) {
+                            // Column name
+                            var k = jexcel.getColumnNameFromId([i, j]);
+                            // Value
+                            data[k] = v;
+                        }
+                    }
+                }
+    
+               return data;
+            } else {
+                cell = jexcel.getIdFromColumnName(cell, true);
+    
+                return key ? obj.records[cell[1]][cell[0]].style[key] : obj.records[cell[1]][cell[0]].getAttribute('style');
+            }
+        },
+    
+        obj.resetStyle = function(o, ignoreHistoryAndEvents) {
+            var keys = Object.keys(o);
+            for (var i = 0; i < keys.length; i++) {
+                // Position
+                var cell = jexcel.getIdFromColumnName(keys[i], true);
+                if (obj.records[cell[1]] && obj.records[cell[1]][cell[0]]) {
+                    obj.records[cell[1]][cell[0]].setAttribute('style', '');
+                }
+            }
+            obj.setStyle(o, null, null, null, ignoreHistoryAndEvents);
+        }
+    
+        /**
+         * Set meta information to cell(s)
+         * 
+         * @return integer
+         */
+        obj.setStyle = function(o, k, v, force, ignoreHistoryAndEvents) {
+            var newValue = {};
+            var oldValue = {};
+    
+            // Apply style
+            var applyStyle = function(cellId, key, value) {
+                // Position
+                var cell = jexcel.getIdFromColumnName(cellId, true);
+    
+                if (obj.records[cell[1]] && obj.records[cell[1]][cell[0]]) {
+                    // Current value
+                    var currentValue = obj.records[cell[1]][cell[0]].style[key];
+    
+                    // Change layout
+                    if (currentValue == value && ! force) {
+                        value = '';
+                        obj.records[cell[1]][cell[0]].style[key] = '';
+                    } else {
+                        obj.records[cell[1]][cell[0]].style[key] = value;
+                    }
+    
+                    // History
+                    if (! oldValue[cellId]) {
+                        oldValue[cellId] = [];
+                    }
+                    if (! newValue[cellId]) {
+                        newValue[cellId] = [];
+                    }
+    
+                    oldValue[cellId].push([key + ':' + currentValue]);
+                    newValue[cellId].push([key + ':' + value]);
+                }
+            }
+    
+            if (k && v) {
+                // Get object from string
+                if (typeof(o) == 'string') {
+                    applyStyle(o, k, v);
+                } else {
+                    // Avoid duplications
+                    var oneApplication = [];
+                    // Apply that for all cells
+                    for (var i = 0; i < o.length; i++) {
+                        var x = o[i].getAttribute('data-x');
+                        var y = o[i].getAttribute('data-y');
+                        var cellName = jexcel.getColumnNameFromId([x, y]);
+                        // This happens when is a merged cell
+                        if (! oneApplication[cellName]) {
+                            applyStyle(cellName, k, v);
+                            oneApplication[cellName] = true;
+                        }
+                    }
+                }
+            } else {
+                var keys = Object.keys(o);
+                for (var i = 0; i < keys.length; i++) {
+                    var style = o[keys[i]];
+                    if (typeof(style) == 'string') {
+                        style = style.split(';');
+                    }
+                    for (var j = 0; j < style.length; j++) {
+                        if (typeof(style[j]) == 'string') {
+                            style[j] = style[j].split(':');
+                        }
+                        // Apply value
+                        if (style[j][0].trim()) {
+                            applyStyle(keys[i], style[j][0].trim(), style[j][1]);
+                        }
+                    }
+                }
+            }
+    
+            var keys = Object.keys(oldValue);
+            for (var i = 0; i < keys.length; i++) {
+                oldValue[keys[i]] = oldValue[keys[i]].join(';');
+            }
+            var keys = Object.keys(newValue);
+            for (var i = 0; i < keys.length; i++) {
+                newValue[keys[i]] = newValue[keys[i]].join(';');
+            }
+    
+            if (! ignoreHistoryAndEvents) {
+                // Keeping history of changes
+                obj.setHistory({
+                    action: 'setStyle',
+                    oldValue: oldValue,
+                    newValue: newValue,
+                });
+            }
+
+            obj.dispatch('onchangestyle', el, o, k, v);
+        }
+
+        /**
+         * Get cell comments, null cell for all
+         */
+        obj.getComments = function(cell, withAuthor) {
+            if (cell) {
+                if (typeof(cell) == 'string') {
+                    var cell = jexcel.getIdFromColumnName(cell, true);
+                }
+
+                if (withAuthor) {
+                    return [obj.records[cell[1]][cell[0]].getAttribute('title'), obj.records[cell[1]][cell[0]].getAttribute('author')];
+                } else {
+                    return obj.records[cell[1]][cell[0]].getAttribute('title') || '';
+                }
+            } else {
+                var data = {};
+                for (var j = 0; j < obj.options.data.length; j++) {
+                    for (var i = 0; i < obj.options.columns.length; i++) {
+                        var comments = obj.records[j][i].getAttribute('title');
+                        if (comments) {
+                            var cell = jexcel.getColumnNameFromId([i, j]);
+                            data[cell] = comments;
+                        }
+                    }
+                }
+                return data;
+            }
+        }
+
+        /**
+         * Set cell comments
+         */
+        obj.setComments = function(cellId, comments, author) {
+            if (typeof(cellId) == 'string') {
+                var cell = jexcel.getIdFromColumnName(cellId, true);
+            } else {
+                var cell = cellId;
+            }
+    
+            // Keep old value
+            var title = obj.records[cell[1]][cell[0]].getAttribute('title');
+            var author = obj.records[cell[1]][cell[0]].getAttribute('data-author');
+            var oldValue = [ title, author ];
+    
+            // Set new values
+            obj.records[cell[1]][cell[0]].setAttribute('title', comments ? comments : '');
+            obj.records[cell[1]][cell[0]].setAttribute('data-author', author ? author : '');
+    
+            // Remove class if there is no comment
+            if (comments) {
+                obj.records[cell[1]][cell[0]].classList.add('jexcel_comments');
+            } else {
+                obj.records[cell[1]][cell[0]].classList.remove('jexcel_comments');
+            }
+    
+            // Save history
+            obj.setHistory({
+                action:'setComments',
+                column: cellId,
+                newValue: [ comments, author ],
+                oldValue: oldValue,
+            });
+        }
+    
+        /**
+         * Get table config information
+         */
+        obj.getConfig = function() {
+            var options = obj.options;
+            options.style = obj.getStyle();
+            options.mergeCells = obj.getMerge();
+            options.comments = obj.getComments();
+    
+            return options;
+        }
+    
+        /**
+         * Sort data and reload table
+         */
+        obj.orderBy = function(column, order) {
+            if (column >= 0) {
+                // Merged cells
+                if (Object.keys(obj.options.mergeCells).length > 0) {
+                    if (! confirm(obj.options.text.thisActionWillDestroyAnyExistingMergedCellsAreYouSure)) {
+                        return false;
+                    } else {
+                        // Remove merged cells
+                        obj.destroyMerged();
+                    }
+                }
+    
+                // Direction
+                if (order == null) {
+                    order = obj.headers[column].classList.contains('arrow-down') ? 1 : 0;
+                } else {
+                    order = order ? 1 : 0;
+                }
+    
+                // Filter
+                Array.prototype.orderBy = function(p, o) {
+                    return this.slice(0).sort(function(a, b) {
+                        var valueA = a[p];
+                        var valueB = b[p];
+    
+                        if (! o) {
+                            return (valueA == '' && valueB != '') ? 1 : (valueA != '' && valueB == '') ? -1 : (valueA > valueB) ? 1 : (valueA < valueB) ? -1 :  0;
+                        } else {
+                            return (valueA == '' && valueB != '') ? 1 : (valueA != '' && valueB == '') ? -1 : (valueA > valueB) ? -1 : (valueA < valueB) ? 1 :  0;
+                        }
+                    });
+                }
+    
+                // Test order
+                var temp = [];
+                if (obj.options.columns[column].type == 'number' || obj.options.columns[column].type == 'percentage' || obj.options.columns[column].type == 'autonumber' || obj.options.columns[column].type == 'color') {
+                    for (var j = 0; j < obj.options.data.length; j++) {
+                        temp[j] = [ j, Number(obj.options.data[j][column]) ];
+                    }
+                } else if (obj.options.columns[column].type == 'calendar' || obj.options.columns[column].type == 'checkbox' || obj.options.columns[column].type == 'radio') {
+                    for (var j = 0; j < obj.options.data.length; j++) {
+                        temp[j] = [ j, obj.options.data[j][column] ];
+                    }
+                } else {
+                    for (var j = 0; j < obj.options.data.length; j++) {
+                        temp[j] = [ j, obj.records[j][column].innerText.toLowerCase() ];
+                    }
+                }
+                temp = temp.orderBy(1, order);
+    
+                // Save history
+                var newValue = [];
+                for (var j = 0; j < temp.length; j++) {
+                    newValue[j] = temp[j][0];
+                }
+    
+                // Save history
+                obj.setHistory({
+                    action: 'orderBy',
+                    rows: newValue,
+                    column: column,
+                    order: order,
+                });
+    
+                // Update order
+                obj.updateOrderArrow(column, order);
+                obj.updateOrder(newValue);
+    
+                // On sort event
+                obj.dispatch('onsort', el, column, order);
+    
+                return true;
+            }
+        }
+    
+        /**
+         * Update order arrow
+         */
+        obj.updateOrderArrow = function(column, order) {
+            // Remove order
+            for (var i = 0; i < obj.headers.length; i++) {
+                obj.headers[i].classList.remove('arrow-up');
+                obj.headers[i].classList.remove('arrow-down');
+            }
+    
+            // No order specified then toggle order
+            if (order) {
+                obj.headers[column].classList.add('arrow-up');
+            } else {
+                obj.headers[column].classList.add('arrow-down');
+            }
+        }
+    
+        /**
+         * Update rows position
+         */
+        obj.updateOrder = function(rows) {
+            // History
+            var data = []
+            for (var j = 0; j < rows.length; j++) {
+                data[j] = obj.options.data[rows[j]];
+            }
+            obj.options.data = data;
+    
+            var data = []
+            for (var j = 0; j < rows.length; j++) {
+                data[j] = obj.records[rows[j]];
+            }
+            obj.records = data;
+    
+            var data = []
+            for (var j = 0; j < rows.length; j++) {
+                data[j] = obj.rows[rows[j]];
+            }
+            obj.rows = data;
+    
+            // Update references
+            obj.updateTableReferences();
+    
+            // Redo search
+            if (obj.results && obj.results.length) {
+                if (obj.searchInput.value) {
+                    obj.search(obj.searchInput.value);
+                } else {
+                    obj.closeFilter();
+                }
+            } else {
+                // Create page
+                obj.results = null;
+                obj.pageNumber = 0;
+    
+                if (obj.options.pagination > 0) {
+                    obj.page(0);
+                } else if (obj.options.lazyLoading == true) {
+                    obj.loadPage(0);
+                } else {
+                    for (var j = 0; j < obj.rows.length; j++) {
+                        obj.tbody.appendChild(obj.rows[j]);
+                    }
+                }
+            }
+        }
+    
+        /**
+         * Move row
+         * 
+         * @return void
+         */
+        obj.moveRow = function(o, d, ignoreDom) {
+            if (Object.keys(obj.options.mergeCells).length > 0) {
+                if (o > d) {
+                    var insertBefore = 1;
+                } else {
+                    var insertBefore = 0;
+                }
+
+                if (obj.isRowMerged(o).length || obj.isRowMerged(d, insertBefore).length) {
+                    if (! confirm(obj.options.text.thisActionWillDestroyAnyExistingMergedCellsAreYouSure)) {
+                        return false;
+                    } else {
+                        obj.destroyMerged();
+                    }
+                }
+            }
+    
+            if (obj.options.search == true) {
+                if (obj.results && obj.results.length != obj.rows.length) {
+                    if (confirm(obj.options.text.thisActionWillClearYourSearchResultsAreYouSure)) {
+                        obj.resetSearch();
+                    } else {
+                        return false;
+                    }
+                }
+    
+                obj.results = null;
+            }
+    
+            if (! ignoreDom) {
+                if (Array.prototype.indexOf.call(obj.tbody.children, obj.rows[d]) >= 0) {
+                    if (o > d) {
+                        obj.tbody.insertBefore(obj.rows[o], obj.rows[d]);
+                    } else {
+                        obj.tbody.insertBefore(obj.rows[o], obj.rows[d].nextSibling);
+                    }
+                } else {
+                    obj.tbody.removeChild(obj.rows[o]);
+                }
+            }
+    
+            // Place references in the correct position
+            obj.rows.splice(d, 0, obj.rows.splice(o, 1)[0]);
+            obj.records.splice(d, 0, obj.records.splice(o, 1)[0]);
+            obj.options.data.splice(d, 0, obj.options.data.splice(o, 1)[0]);
+    
+            // Respect pagination
+            if (obj.options.pagination > 0 && obj.tbody.children.length != obj.options.pagination) {
+                obj.page(obj.pageNumber);
+            }
+    
+            // Keeping history of changes
+            obj.setHistory({
+                action:'moveRow',
+                oldValue: o,
+                newValue: d,
+            });
+    
+            // Update table references
+            obj.updateTableReferences();
+    
+            // Events
+            obj.dispatch('onmoverow', el, o, d);
+        }
+
+        /**
+         * Insert a new row
+         * 
+         * @param mixed - number of blank lines to be insert or a single array with the data of the new row
+         * @param rowNumber
+         * @param insertBefore
+         * @return void
+         */
+        obj.insertRow = function(mixed, rowNumber, insertBefore) {
+            // Configuration
+            if (obj.options.allowInsertRow == true) {
+                // Records
+                var records = [];
+    
+                // Data to be insert
+                var data = [];
+    
+                // The insert could be lead by number of rows or the array of data
+                if (mixed > 0) {
+                    var numOfRows = mixed;
+                } else {
+                    var numOfRows = 1;
+    
+                    if (mixed) {
+                        data = mixed;
+                    }
+                }
+    
+                // Direction
+                var insertBefore = insertBefore ? true : false;
+    
+                // Current column number
+                var lastRow = obj.options.data.length - 1;
+    
+                if (rowNumber == undefined || rowNumber >= parseInt(lastRow) || rowNumber < 0) {
+                    rowNumber = lastRow;
+                }
+    
+                // Onbeforeinsertrow
+                if (obj.dispatch('onbeforeinsertrow', el, rowNumber, numOfRows, insertBefore) === false) {
+                    console.log('onbeforeinsertrow returned false');
+
+                    return false;
+                }
+    
+                // Merged cells
+                if (Object.keys(obj.options.mergeCells).length > 0) {
+                    if (obj.isRowMerged(rowNumber, insertBefore).length) {
+                        if (! confirm(obj.options.text.thisActionWillDestroyAnyExistingMergedCellsAreYouSure)) {
+                            return false;
+                        } else {
+                            obj.destroyMerged();
+                        }
+                    }
+                }
+    
+                // Clear any search
+                if (obj.options.search == true) {
+                    if (obj.results && obj.results.length != obj.rows.length) {
+                        if (confirm(obj.options.text.thisActionWillClearYourSearchResultsAreYouSure)) {
+                            obj.resetSearch();
+                        } else {
+                            return false;
+                        }
+                    }
+    
+                    obj.results = null;
+                }
+    
+                // Insertbefore
+                var rowIndex = (! insertBefore) ? rowNumber + 1 : rowNumber;
+
+                // Keep the current data
+                var currentRecords = obj.records.splice(rowIndex);
+                var currentData = obj.options.data.splice(rowIndex);
+                var currentRows = obj.rows.splice(rowIndex);
+    
+                // Adding lines
+                var rowRecords = [];
+                var rowData = [];
+                var rowNode = [];
+
+                for (var row = rowIndex; row < (numOfRows + rowIndex); row++) {
+                    // Push data to the data container
+                    obj.options.data[row] = [];
+                    for (var col = 0; col < obj.options.columns.length; col++) {
+                        obj.options.data[row][col]  = data[col] ? data[col] : '';
+                    }
+                    // Create row
+                    var tr = obj.createRow(row, obj.options.data[row]);
+                    // Append node
+                    if (currentRows[0]) {
+                        if (Array.prototype.indexOf.call(obj.tbody.children, currentRows[0]) >= 0) {
+                            obj.tbody.insertBefore(tr, currentRows[0]);
+                        }
+                    } else {
+                        if (Array.prototype.indexOf.call(obj.tbody.children, obj.rows[rowNumber]) >= 0) {
+                            obj.tbody.appendChild(tr);
+                        }
+                    }
+                    // Record History
+                    rowRecords.push(obj.records[row]);
+                    rowData.push(obj.options.data[row]);
+                    rowNode.push(tr);
+                }
+    
+                // Copy the data back to the main data
+                Array.prototype.push.apply(obj.records, currentRecords);
+                Array.prototype.push.apply(obj.options.data, currentData);
+                Array.prototype.push.apply(obj.rows, currentRows);
+    
+                // Respect pagination
+                if (obj.options.pagination > 0) {
+                    obj.page(obj.pageNumber);
+                }
+    
+                // Keep history
+                obj.setHistory({
+                    action: 'insertRow',
+                    rowNumber: rowNumber,
+                    numOfRows: numOfRows,
+                    insertBefore: insertBefore,
+                    rowRecords: rowRecords,
+                    rowData: rowData,
+                    rowNode: rowNode,
+                });
+    
+                // Remove table references
+                obj.updateTableReferences();
+    
+                // Events
+                obj.dispatch('oninsertrow', el, rowNumber, numOfRows, rowRecords, insertBefore);
+            }
+        }
+    
+        /**
+         * Delete a row by number
+         * 
+         * @param integer rowNumber - row number to be excluded
+         * @param integer numOfRows - number of lines
+         * @return void
+         */
+        obj.deleteRow = function(rowNumber, numOfRows) {
+            // Global Configuration
+            if (obj.options.allowDeleteRow == true) {
+                if (obj.options.allowDeletingAllRows == true || obj.options.data.length > 1) {
+                    // Delete row definitions
+                    if (rowNumber == undefined) {
+                        var number = obj.getSelectedRows();
+    
+                        if (! number[0]) {
+                            rowNumber = obj.options.data.length - 1;
+                            numOfRows = 1;
+                        } else {
+                            rowNumber = parseInt(number[0].getAttribute('data-y'));
+                            numOfRows = number.length;
+                        }
+                    }
+    
+                    // Last column
+                    var lastRow = obj.options.data.length - 1;
+    
+                    if (rowNumber == undefined || rowNumber > lastRow || rowNumber < 0) {
+                        rowNumber = lastRow;
+                    }
+    
+                    if (! numOfRows) {
+                        numOfRows = 1;
+                    }
+    
+                    // Do not delete more than the number of recoreds
+                    if (rowNumber + numOfRows >= obj.options.data.length) {
+                        numOfRows = obj.options.data.length - rowNumber;
+                    }
+
+                    // Onbeforedeleterow
+                    if (obj.dispatch('onbeforedeleterow', el, rowNumber, numOfRows) === false) {
+                        console.log('onbeforedeleterow returned false');
+                        return false;
+                    }
+    
+                    if (parseInt(rowNumber) > -1) {
+                        // Merged cells
+                        var mergeExists = false;
+                        if (Object.keys(obj.options.mergeCells).length > 0) {
+                            for (var row = rowNumber; row < rowNumber + numOfRows; row++) {
+                                if (obj.isRowMerged(row, false).length) {
+                                    mergeExists = true;
+                                }
+                            }
+                        }
+                        if (mergeExists) {
+                            if (! confirm(obj.options.text.thisActionWillDestroyAnyExistingMergedCellsAreYouSure)) {
+                                return false;
+                            } else {
+                                obj.destroyMerged();
+                            }
+                        }
+    
+                        // Clear any search
+                        if (obj.options.search == true) {
+                            if (obj.results && obj.results.length != obj.rows.length) {
+                                if (confirm(obj.options.text.thisActionWillClearYourSearchResultsAreYouSure)) {
+                                    obj.resetSearch();
+                                } else {
+                                    return false;
+                                }
+                            }
+    
+                            obj.results = null;
+                        }
+    
+                        // Remove node
+                        for (var row = rowNumber; row < rowNumber + numOfRows; row++) {
+                            if (Array.prototype.indexOf.call(obj.tbody.children, obj.rows[row]) >= 0) {
+                                obj.rows[row].className = '';
+                                obj.rows[row].parentNode.removeChild(obj.rows[row]);
+                            }
+                        }
+    
+                        // Remove data
+                        var rowRecords = obj.records.splice(rowNumber, numOfRows);
+                        var rowData = obj.options.data.splice(rowNumber, numOfRows);
+                        var rowNode = obj.rows.splice(rowNumber, numOfRows);
+    
+                        // Respect pagination
+                        if (obj.options.pagination > 0 && obj.tbody.children.length != obj.options.pagination) {
+                            obj.page(obj.pageNumber);
+                        }
+    
+                        // Remove selection
+                        obj.conditionalSelectionUpdate(1, rowNumber, (rowNumber + numOfRows) - 1);
+    
+                        // Keep history
+                        obj.setHistory({
+                            action: 'deleteRow',
+                            rowNumber: rowNumber,
+                            numOfRows: numOfRows,
+                            insertBefore: 1,
+                            rowRecords: rowRecords,
+                            rowData: rowData,
+                            rowNode: rowNode
+                        });
+    
+                        // Remove table references
+                        obj.updateTableReferences();
+    
+                        // Events
+                        obj.dispatch('ondeleterow', el, rowNumber, numOfRows, rowRecords);
+                    }
+                } else {
+                    console.error('JEXCEL. It is not possible to delete the last row');
+                }
+            }
+        }
+    
+    
+        /**
+         * Move column
+         * 
+         * @return void
+         */
+        obj.moveColumn = function(o, d) {
+            if (Object.keys(obj.options.mergeCells).length > 0) {
+                if (o > d) {
+                    var insertBefore = 1;
+                } else {
+                    var insertBefore = 0;
+                }
+
+                if (obj.isColMerged(o).length || obj.isColMerged(d, insertBefore).length) {
+                    if (! confirm(obj.options.text.thisActionWillDestroyAnyExistingMergedCellsAreYouSure)) {
+                        return false;
+                    } else {
+                        obj.destroyMerged();
+                    }
+                }
+            }
+    
+            var o = parseInt(o);
+            var d = parseInt(d);
+    
+            if (o > d) {
+                obj.headerContainer.insertBefore(obj.headers[o], obj.headers[d]);
+                obj.colgroupContainer.insertBefore(obj.colgroup[o], obj.colgroup[d]);
+    
+                for (var j = 0; j < obj.rows.length; j++) {
+                    obj.rows[j].insertBefore(obj.records[j][o], obj.records[j][d]);
+                }
+            } else {
+                obj.headerContainer.insertBefore(obj.headers[o], obj.headers[d].nextSibling);
+                obj.colgroupContainer.insertBefore(obj.colgroup[o], obj.colgroup[d].nextSibling);
+    
+                for (var j = 0; j < obj.rows.length; j++) {
+                    obj.rows[j].insertBefore(obj.records[j][o], obj.records[j][d].nextSibling);
+                }
+            }
+    
+            obj.options.columns.splice(d, 0, obj.options.columns.splice(o, 1)[0]);
+            obj.headers.splice(d, 0, obj.headers.splice(o, 1)[0]);
+            obj.colgroup.splice(d, 0, obj.colgroup.splice(o, 1)[0]);
+    
+            for (var j = 0; j < obj.rows.length; j++) {
+                obj.options.data[j].splice(d, 0, obj.options.data[j].splice(o, 1)[0]);
+                obj.records[j].splice(d, 0, obj.records[j].splice(o, 1)[0]);
+            }
+    
+            // Update footers position
+            if (obj.options.footers) {
+                for (var j = 0; j < obj.options.footers.length; j++) {
+                    obj.options.footers[j].splice(d, 0, obj.options.footers[j].splice(o, 1)[0]);
+                }
+            }
+
+            // Keeping history of changes
+            obj.setHistory({
+                action:'moveColumn',
+                oldValue: o,
+                newValue: d,
+            });
+    
+            // Update table references
+            obj.updateTableReferences();
+    
+            // Events
+            obj.dispatch('onmovecolumn', el, o, d);
+        }
+
+        /**
+         * Insert a new column
+         * 
+         * @param mixed - num of columns to be added or data to be added in one single column
+         * @param int columnNumber - number of columns to be created
+         * @param bool insertBefore
+         * @param object properties - column properties
+         * @return void
+         */
+        obj.insertColumn = function(mixed, columnNumber, insertBefore, properties) {
+            // Configuration
+            if (obj.options.allowInsertColumn == true) {
+                // Records
+                var records = [];
+    
+                // Data to be insert
+                var data = [];
+    
+                // The insert could be lead by number of rows or the array of data
+                if (mixed > 0) {
+                    var numOfColumns = mixed;
+                } else {
+                    var numOfColumns = 1;
+    
+                    if (mixed) {
+                        data = mixed;
+                    }
+                }
+    
+                // Direction
+                var insertBefore = insertBefore ? true : false;
+    
+                // Current column number
+                var lastColumn = obj.options.columns.length - 1;
+    
+                // Confirm position
+                if (columnNumber == undefined || columnNumber >= parseInt(lastColumn) || columnNumber < 0) {
+                    columnNumber = lastColumn;
+                }
+    
+                // Onbeforeinsertcolumn
+                if (obj.dispatch('onbeforeinsertcolumn', el, columnNumber, numOfColumns, insertBefore) === false) {
+                    console.log('onbeforeinsertcolumn returned false');
+
+                    return false;
+                }
+    
+                // Merged cells
+                if (Object.keys(obj.options.mergeCells).length > 0) {
+                    if (obj.isColMerged(columnNumber, insertBefore).length) {
+                        if (! confirm(obj.options.text.thisActionWillDestroyAnyExistingMergedCellsAreYouSure)) {
+                            return false;
+                        } else {
+                            obj.destroyMerged();
+                        }
+                    }
+                }
+    
+                // Create default properties
+                if (! properties) {
+                    properties = [];
+                }
+    
+                for (var i = 0; i < numOfColumns; i++) {
+                    if (! properties[i]) {
+                        properties[i] = { type:'text', source:[], options:[], width:obj.options.defaultColWidth, align:obj.options.defaultColAlign };
+                    }
+                }
+    
+                // Insert before
+                var columnIndex = (! insertBefore) ? columnNumber + 1 : columnNumber;
+                obj.options.columns = jexcel.injectArray(obj.options.columns, columnIndex, properties);
+    
+                // Open space in the containers
+                var currentHeaders = obj.headers.splice(columnIndex);
+                var currentColgroup = obj.colgroup.splice(columnIndex);
+    
+                // History
+                var historyHeaders = [];
+                var historyColgroup = [];
+                var historyRecords = [];
+                var historyData = [];
+                var historyFooters = [];
+    
+                // Add new headers
+                for (var col = columnIndex; col < (numOfColumns + columnIndex); col++) {
+                    obj.createCellHeader(col);
+                    obj.headerContainer.insertBefore(obj.headers[col], obj.headerContainer.children[col+1]);
+                    obj.colgroupContainer.insertBefore(obj.colgroup[col], obj.colgroupContainer.children[col+1]);
+    
+                    historyHeaders.push(obj.headers[col]);
+                    historyColgroup.push(obj.colgroup[col]);
+                }
+    
+                // Add new footer cells
+                if (obj.options.footers) {
+                    for (var j = 0; j < obj.options.footers.length; j++) {
+                        historyFooters[j] = [];
+                        for (var i = 0; i < numOfColumns; i++) {
+                            historyFooters[j].push('');
+                        }
+                        obj.options.footers[j].splice(columnIndex, 0, historyFooters[j]);
+                    }
+                }
+
+                // Adding visual columns
+                for (var row = 0; row < obj.options.data.length; row++) {
+                    // Keep the current data
+                    var currentData = obj.options.data[row].splice(columnIndex);
+                    var currentRecord = obj.records[row].splice(columnIndex);
+    
+                    // History
+                    historyData[row] = [];
+                    historyRecords[row] = [];
+    
+                    for (var col = columnIndex; col < (numOfColumns + columnIndex); col++) {
+                        // New value
+                        var value = data[row] ? data[row] : '';
+                        obj.options.data[row][col] = value;
+                        // New cell
+                        var td = obj.createCell(col, row, obj.options.data[row][col]);
+                        obj.records[row][col] = td;
+                        // Add cell to the row
+                        if (obj.rows[row]) {
+                            obj.rows[row].insertBefore(td, obj.rows[row].children[col+1]);
+                        }
+    
+                        // Record History
+                        historyData[row].push(value);
+                        historyRecords[row].push(td);
+                    }
+    
+                    // Copy the data back to the main data
+                    Array.prototype.push.apply(obj.options.data[row], currentData);
+                    Array.prototype.push.apply(obj.records[row], currentRecord);
+                }
+    
+                Array.prototype.push.apply(obj.headers, currentHeaders);
+                Array.prototype.push.apply(obj.colgroup, currentColgroup);
+    
+                // Adjust nested headers
+                if (obj.options.nestedHeaders && obj.options.nestedHeaders.length > 0) {
+                    // Flexible way to handle nestedheaders
+                    if (obj.options.nestedHeaders[0] && obj.options.nestedHeaders[0][0]) {
+                        for (var j = 0; j < obj.options.nestedHeaders.length; j++) {
+                            var colspan = parseInt(obj.options.nestedHeaders[j][obj.options.nestedHeaders[j].length-1].colspan) + numOfColumns;
+                            obj.options.nestedHeaders[j][obj.options.nestedHeaders[j].length-1].colspan = colspan;
+                            obj.thead.children[j].children[obj.thead.children[j].children.length-1].setAttribute('colspan', colspan);
+                        }
+                    } else {
+                        var colspan = parseInt(obj.options.nestedHeaders[0].colspan) + numOfColumns;
+                        obj.options.nestedHeaders[0].colspan = colspan;
+                        obj.thead.children[0].children[obj.thead.children[0].children.length-1].setAttribute('colspan', colspan);
+                    }
+                }
+    
+                // Keep history
+                obj.setHistory({
+                    action: 'insertColumn',
+                    columnNumber:columnNumber,
+                    numOfColumns:numOfColumns,
+                    insertBefore:insertBefore,
+                    columns:properties,
+                    headers:historyHeaders,
+                    colgroup:historyColgroup,
+                    records:historyRecords,
+                    footers:historyFooters,
+                    data:historyData,
+                });
+    
+                // Remove table references
+                obj.updateTableReferences();
+    
+                // Events
+                obj.dispatch('oninsertcolumn', el, columnNumber, numOfColumns, historyRecords, insertBefore);
+
+                obj.el.querySelector("table.jexcel tr").childNodes.forEach(function (td) {
+                  td.style.textAlign = "center";
+                });
+            }
+        }
+    
+        /**
+         * Delete a column by number
+         * 
+         * @param integer columnNumber - reference column to be excluded
+         * @param integer numOfColumns - number of columns to be excluded from the reference column
+         * @return void
+         */
+        obj.deleteColumn = function(columnNumber, numOfColumns) {
+            // Global Configuration
+            if (obj.options.allowDeleteColumn == true) {
+                if (obj.headers.length > 1) {
+                    // Delete column definitions
+                    if (columnNumber == undefined) {
+                        var number = obj.getSelectedColumns(true);
+    
+                        if (! number.length) {
+                            // Remove last column
+                            columnNumber = obj.headers.length - 1;
+                            numOfColumns = 1;
+                        } else {
+                            // Remove selected
+                            columnNumber = parseInt(number[0]);
+                            numOfColumns = parseInt(number.length);
+                        }
+                    }
+    
+                    // Lasat column
+                    var lastColumn = obj.options.data[0].length - 1;
+    
+                    if (columnNumber == undefined || columnNumber > lastColumn || columnNumber < 0) {
+                        columnNumber = lastColumn;
+                    }
+    
+                    // Minimum of columns to be delete is 1
+                    if (! numOfColumns) {
+                        numOfColumns = 1;
+                    }
+    
+                    // Can't delete more than the limit of the table
+                    if (numOfColumns > obj.options.data[0].length - columnNumber) {
+                        numOfColumns = obj.options.data[0].length - columnNumber;
+                    }
+    
+                    // onbeforedeletecolumn
+                   if (obj.dispatch('onbeforedeletecolumn', el, columnNumber, numOfColumns) === false) {
+                      console.log('onbeforedeletecolumn returned false');
+                      return false;
+                   }
+    
+                    // Can't remove the last column
+                    if (parseInt(columnNumber) > -1) {
+                        // Merged cells
+                        var mergeExists = false;
+                        if (Object.keys(obj.options.mergeCells).length > 0) {
+                            for (var col = columnNumber; col < columnNumber + numOfColumns; col++) {
+                                if (obj.isColMerged(col, false).length) {
+                                    mergeExists = true;
+                                }
+                            }
+                        }
+                        if (mergeExists) {
+                            if (! confirm(obj.options.text.thisActionWillDestroyAnyExistingMergedCellsAreYouSure)) {
+                                return false;
+                            } else {
+                                obj.destroyMerged();
+                            }
+                        }
+    
+                        // Delete the column properties
+                        var columns = obj.options.columns.splice(columnNumber, numOfColumns);
+    
+                        for (var col = columnNumber; col < columnNumber + numOfColumns; col++) {
+                            obj.colgroup[col].className = '';
+                            obj.headers[col].className = '';
+                            obj.colgroup[col].parentNode.removeChild(obj.colgroup[col]);
+                            obj.headers[col].parentNode.removeChild(obj.headers[col]);
+                        }
+    
+                        var historyHeaders = obj.headers.splice(columnNumber, numOfColumns);
+                        var historyColgroup = obj.colgroup.splice(columnNumber, numOfColumns);
+                        var historyRecords = [];
+                        var historyData = [];
+                        var historyFooters = [];
+
+                        for (var row = 0; row < obj.options.data.length; row++) {
+                            for (var col = columnNumber; col < columnNumber + numOfColumns; col++) {
+                                obj.records[row][col].className = '';
+                                obj.records[row][col].parentNode.removeChild(obj.records[row][col]);
+                            }
+                        }
+    
+                        // Delete headers
+                        for (var row = 0; row < obj.options.data.length; row++) {
+                            // History
+                            historyData[row] = obj.options.data[row].splice(columnNumber, numOfColumns);
+                            historyRecords[row] = obj.records[row].splice(columnNumber, numOfColumns);
+                        }
+
+                        // Delete footers
+                        if (obj.options.footers) {
+                            for (var row = 0; row < obj.options.footers.length; row++) {
+                                historyFooters[row] = obj.options.footers[row].splice(columnNumber, numOfColumns);
+                            }
+                        }
+
+                        // Remove selection
+                        obj.conditionalSelectionUpdate(0, columnNumber, (columnNumber + numOfColumns) - 1);
+    
+                        // Adjust nested headers
+                        if (obj.options.nestedHeaders && obj.options.nestedHeaders.length > 0) {
+                            // Flexible way to handle nestedheaders
+                            if (obj.options.nestedHeaders[0] && obj.options.nestedHeaders[0][0]) {
+                                for (var j = 0; j < obj.options.nestedHeaders.length; j++) {
+                                    var colspan = parseInt(obj.options.nestedHeaders[j][obj.options.nestedHeaders[j].length-1].colspan) - numOfColumns;
+                                    obj.options.nestedHeaders[j][obj.options.nestedHeaders[j].length-1].colspan = colspan;
+                                    obj.thead.children[j].children[obj.thead.children[j].children.length-1].setAttribute('colspan', colspan);
+                                }
+                            } else {
+                                var colspan = parseInt(obj.options.nestedHeaders[0].colspan) - numOfColumns;
+                                obj.options.nestedHeaders[0].colspan = colspan;
+                                obj.thead.children[0].children[obj.thead.children[0].children.length-1].setAttribute('colspan', colspan);
+                            }
+                        }
+    
+                        // Keeping history of changes
+                        obj.setHistory({
+                            action:'deleteColumn',
+                            columnNumber:columnNumber,
+                            numOfColumns:numOfColumns,
+                            insertBefore: 1,
+                            columns:columns,
+                            headers:historyHeaders,
+                            colgroup:historyColgroup,
+                            records:historyRecords,
+                            footers:historyFooters,
+                            data:historyData,
+                        });
+    
+                        // Update table references
+                        obj.updateTableReferences();
+    
+                        // Delete
+                        obj.dispatch('ondeletecolumn', el, columnNumber, numOfColumns, historyRecords);
+                    }
+                } else {
+                    console.error('JEXCEL. It is not possible to delete the last column');
+                }
+            }
+        }
+    
+        /**
+         * Get seleted rows numbers
+         * 
+         * @return array
+         */
+        obj.getSelectedRows = function(asIds) {
+            var rows = [];
+            // Get all selected rows
+            for (var j = 0; j < obj.rows.length; j++) {
+                if (obj.rows[j].classList.contains('selected')) {
+                    if (asIds) {
+                        rows.push(j);
+                    } else {
+                        rows.push(obj.rows[j]);
+                    }
+                }
+            }
+    
+            return rows;
+        },
+    
+        /**
+         * Get seleted column numbers
+         * 
+         * @return array
+         */
+        obj.getSelectedColumns = function() {
+            var cols = [];
+            // Get all selected cols
+            for (var i = 0; i < obj.headers.length; i++) {
+                if (obj.headers[i].classList.contains('selected')) {
+                    cols.push(i);
+                }
+            }
+    
+            return cols;
+        }
+    
+        /**
+         * Get highlighted
+         * 
+         * @return array
+         */
+        obj.getHighlighted = function() {
+            return obj.highlighted;
+        }
+    
+        /**
+         * Update cell references
+         * 
+         * @return void
+         */
+        obj.updateTableReferences = function() {
+            // Update headers
+            for (var i = 0; i < obj.headers.length; i++) {
+                var x = obj.headers[i].getAttribute('data-x');
+    
+                if (x != i) {
+                    // Update coords
+                    obj.headers[i].setAttribute('data-x', i);
+                    // Title
+                    if (! obj.headers[i].getAttribute('title')) {
+                        obj.headers[i].innerHTML = jexcel.getColumnName(i);
+                    }
+                }
+            }
+    
+            // Update all rows
+            for (var j = 0; j < obj.rows.length; j++) {
+                if (obj.rows[j]) {
+                    var y = obj.rows[j].getAttribute('data-y');
+    
+                    if (y != j) {
+                        // Update coords
+                        obj.rows[j].setAttribute('data-y', j);
+                        obj.rows[j].children[0].setAttribute('data-y', j);
+                        // Row number
+                        obj.rows[j].children[0].innerHTML = j + 1;
+                    }
+                }
+            }
+    
+            // Regular cells affected by this change
+            var affectedTokens = [];
+            var mergeCellUpdates = [];
+    
+            // Update cell
+            var updatePosition = function(x,y,i,j) {
+                if (x != i) {
+                    obj.records[j][i].setAttribute('data-x', i);
+                }
+                if (y != j) {
+                    obj.records[j][i].setAttribute('data-y', j);
+                }
+    
+                // Other updates
+                if (x != i || y != j) {
+                    var columnIdFrom = jexcel.getColumnNameFromId([x, y]);
+                    var columnIdTo = jexcel.getColumnNameFromId([i, j]);
+                    affectedTokens[columnIdFrom] = columnIdTo;
+                }
+            }
+    
+            for (var j = 0; j < obj.records.length; j++) {
+                for (var i = 0; i < obj.records[0].length; i++) {
+                    if (obj.records[j][i]) {
+                        // Current values
+                        var x = obj.records[j][i].getAttribute('data-x');
+                        var y = obj.records[j][i].getAttribute('data-y');
+    
+                        // Update column
+                        if (obj.records[j][i].getAttribute('data-merged')) {
+                            var columnIdFrom = jexcel.getColumnNameFromId([x, y]);
+                            var columnIdTo = jexcel.getColumnNameFromId([i, j]);
+                            if (mergeCellUpdates[columnIdFrom] == null) {
+                                if (columnIdFrom == columnIdTo) {
+                                    mergeCellUpdates[columnIdFrom] = false;
+                                } else {
+                                    var totalX = parseInt(i - x);
+                                    var totalY = parseInt(j - y);
+                                    mergeCellUpdates[columnIdFrom] = [ columnIdTo, totalX, totalY ];
+                                }
+                            }
+                        } else {
+                            updatePosition(x,y,i,j);
+                        }
+                    }
+                }
+            }
+    
+            // Update merged if applicable
+            var keys = Object.keys(mergeCellUpdates);
+            if (keys.length) {
+                for (var i = 0; i < keys.length; i++) {
+                    if (mergeCellUpdates[keys[i]]) {
+                        var info = jexcel.getIdFromColumnName(keys[i], true)
+                        var x = info[0];
+                        var y = info[1];
+                        updatePosition(x,y,x + mergeCellUpdates[keys[i]][1],y + mergeCellUpdates[keys[i]][2]);
+    
+                        var columnIdFrom = keys[i];
+                        var columnIdTo = mergeCellUpdates[keys[i]][0];
+                        for (var j = 0; j < obj.options.mergeCells[columnIdFrom][2].length; j++) {
+                            var x = parseInt(obj.options.mergeCells[columnIdFrom][2][j].getAttribute('data-x'));
+                            var y = parseInt(obj.options.mergeCells[columnIdFrom][2][j].getAttribute('data-y'));
+                            obj.options.mergeCells[columnIdFrom][2][j].setAttribute('data-x', x + mergeCellUpdates[keys[i]][1]);
+                            obj.options.mergeCells[columnIdFrom][2][j].setAttribute('data-y', y + mergeCellUpdates[keys[i]][2]);
+                        }
+    
+                        obj.options.mergeCells[columnIdTo] = obj.options.mergeCells[columnIdFrom];
+                        delete(obj.options.mergeCells[columnIdFrom]);
+                    }
+                }
+            }
+    
+            // Update formulas
+            obj.updateFormulas(affectedTokens);
+    
+            // Update meta data
+            obj.updateMeta(affectedTokens);
+    
+            // Refresh selection
+            obj.refreshSelection();
+    
+            // Update table with custom configuration if applicable
+            obj.updateTable();
+        }
+    
+        /**
+         * Custom settings for the cells
+         */
+        obj.updateTable = function() {
+            // Check for spare
+            if (obj.options.minSpareRows > 0) {
+                var numBlankRows = 0;
+                for (var j = obj.rows.length - 1; j >= 0; j--) {
+                    var test = false;
+                    for (var i = 0; i < obj.headers.length; i++) {
+                        if (obj.options.data[j][i]) {
+                            test = true;
+                        }
+                    }
+                    if (test) {
+                        break;
+                    } else {
+                        numBlankRows++;
+                    }
+                }
+    
+                if (obj.options.minSpareRows - numBlankRows > 0) {
+                    obj.insertRow(obj.options.minSpareRows - numBlankRows)
+                }
+            }
+    
+            if (obj.options.minSpareCols > 0) {
+                var numBlankCols = 0;
+                for (var i = obj.headers.length - 1; i >= 0 ; i--) {
+                    var test = false;
+                    for (var j = 0; j < obj.rows.length; j++) {
+                        if (obj.options.data[j][i]) {
+                            test = true;
+                        }
+                    }
+                    if (test) {
+                        break;
+                    } else {
+                        numBlankCols++;
+                    }
+                }
+    
+                if (obj.options.minSpareCols - numBlankCols > 0) {
+                    obj.insertColumn(obj.options.minSpareCols - numBlankCols)
+                }
+            }
+    
+            // Customizations by the developer
+            if (typeof(obj.options.updateTable) == 'function') {
+                if (obj.options.detachForUpdates) {
+                    el.removeChild(obj.content);
+                }
+
+                for (var j = 0; j < obj.rows.length; j++) {
+                    for (var i = 0; i < obj.headers.length; i++) {
+                        obj.options.updateTable(el, obj.records[j][i], i, j, obj.options.data[j][i], obj.records[j][i].innerText, jexcel.getColumnNameFromId([i, j]));
+                    }
+                }
+
+                if (obj.options.detachForUpdates) {
+                    el.insertBefore(obj.content, obj.pagination);
+                }
+            }
+    
+            // Update footers
+            if (obj.options.footers) {
+                obj.setFooter();
+            }
+
+            // Update corner position
+            setTimeout(function() {
+                obj.updateCornerPosition();
+            },0);
+        }
+
+        /**
+         * Show row
+         */
+        obj.showRow = function(rowNumber) {
+            obj.rows[rowNumber].style.display = '';
+        }
+
+        /**
+         * Hide row
+         */
+        obj.hideRow = function(rowNumber) {
+            obj.rows[rowNumber].style.display = 'none';
+        }
+
+        /**
+         * Show column
+         */
+        obj.showColumn = function(colNumber) {
+            obj.headers[colNumber].style.display = '';
+            obj.colgroup[colNumber].style.display = '';
+            for (var j = 0; j < obj.options.data.length; j++) {
+                obj.records[j][colNumber].style.display = '';
+            }
+        }
+
+        /**
+         * Hide column
+         */
+        obj.hideColumn = function(colNumber) {
+            obj.headers[colNumber].style.display = 'none';
+            obj.colgroup[colNumber].style.display = 'none';
+            for (var j = 0; j < obj.options.data.length; j++) {
+                obj.records[j][colNumber].style.display = 'none';
+            }
+        }
+
+        /**
+         * Show index column
+         */
+        obj.showIndex = function() {
+            obj.table.classList.remove('jexcel_hidden_index');
+        }
+    
+        /**
+         * Hide index column
+         */
+        obj.hideIndex = function() {
+            obj.table.classList.add('jexcel_hidden_index');
+        }
+    
+        /**
+         * Update all related cells in the chain
+         */
+        var chainLoopProtection = [];
+    
+        obj.updateFormulaChain = function(x, y, records) {
+            var cellId = jexcel.getColumnNameFromId([x, y]);
+            if (obj.formula[cellId] && obj.formula[cellId].length > 0) {
+                if (chainLoopProtection[cellId]) {
+                    obj.records[y][x].innerHTML = '#ERROR';
+                    obj.formula[cellId] = '';
+                } else {
+                    // Protection
+                    chainLoopProtection[cellId] = true;
+    
+                    for (var i = 0; i < obj.formula[cellId].length; i++) {
+                        var cell = jexcel.getIdFromColumnName(obj.formula[cellId][i], true);
+                        // Update cell
+                        var value = ''+obj.options.data[cell[1]][cell[0]];
+                        if (value.substr(0,1) == '=') {
+                            records.push(obj.updateCell(cell[0], cell[1], value, true));
+                        } else {
+                            // No longer a formula, remove from the chain
+                            Object.keys(obj.formula)[i] = null;
+                        }
+                        obj.updateFormulaChain(cell[0], cell[1], records);
+                    }
+                }
+            }
+    
+            chainLoopProtection = [];
+        }
+    
+        /**
+         * Update formulas
+         */
+        obj.updateFormulas = function(referencesToUpdate) {
+            // Update formulas
+            for (var j = 0; j < obj.options.data.length; j++) {
+                for (var i = 0; i < obj.options.data[0].length; i++) {
+                    var value = '' + obj.options.data[j][i];
+                    // Is formula
+                    if (value.substr(0,1) == '=') {
+                        // Replace tokens
+                        var newFormula = obj.updateFormula(value, referencesToUpdate);
+                        if (newFormula != value) {
+                            obj.options.data[j][i] = newFormula;
+                        }
+                    }
+                }
+            }
+    
+            // Update formula chain
+            var formula = [];
+            var keys = Object.keys(obj.formula);
+            for (var j = 0; j < keys.length; j++) {
+                // Current key and values
+                var key = keys[j];
+                var value = obj.formula[key];
+                // Update key
+                if (referencesToUpdate[key]) {
+                    key = referencesToUpdate[key];
+                }
+                // Update values
+                formula[key] = [];
+                for (var i = 0; i < value.length; i++) {
+                    var letter = value[i];
+                    if (referencesToUpdate[letter]) {
+                        letter = referencesToUpdate[letter];
+                    }
+                    formula[key].push(letter);
+                }
+            }
+            obj.formula = formula;
+        }
+    
+        /**
+         * Update formula
+         */
+        obj.updateFormula = function(formula, referencesToUpdate) {
+            var testLetter = /[A-Z]/;
+            var testNumber = /[0-9]/;
+    
+            var newFormula = '';
+            var letter = null;
+            var number = null;
+            var token = '';
+    
+            for (var index = 0; index < formula.length; index++) {
+                if (testLetter.exec(formula[index])) {
+                    letter = 1;
+                    number = 0;
+                    token += formula[index];
+                } else if (testNumber.exec(formula[index])) {
+                    number = letter ? 1 : 0;
+                    token += formula[index];
+                } else {
+                    if (letter && number) {
+                        token = referencesToUpdate[token] ? referencesToUpdate[token] : token;
+                    }
+                    newFormula += token;
+                    newFormula += formula[index];
+                    letter = 0;
+                    number = 0;
+                    token = '';
+                }
+            }
+    
+            if (token) {
+                if (letter && number) {
+                    token = referencesToUpdate[token] ? referencesToUpdate[token] : token;
+                }
+                newFormula += token;
+            }
+    
+            return newFormula;
+        }
+    
+        /**
+         * Secure formula
+         */
+        var secureFormula = function(oldValue) {
+            var newValue = '';
+            var inside = 0;
+
+            for (var i = 0; i < oldValue.length; i++) {
+                if (oldValue[i] == '"') {
+                    if (inside == 0) {
+                        inside = 1;
+                    } else {
+                        inside = 0;
+                    }
+                }
+
+                if (inside == 1) {
+                    newValue += oldValue[i];
+                } else {
+                    newValue += oldValue[i].toUpperCase();
+                }
+            }
+
+            return newValue;
+        }
+
+        /**
+         * Parse formulas
+         */
+        obj.executeFormula = function(expression, x, y) {
+    
+            var formulaResults = [];
+            var formulaLoopProtection = [];
+    
+            // Execute formula with loop protection
+            var execute = function(expression, x, y) {
+             // Parent column identification
+                var parentId = jexcel.getColumnNameFromId([x, y]);
+    
+                // Code protection
+                if (formulaLoopProtection[parentId]) {
+                    console.error('Reference loop detected');
+                    return '#ERROR';
+                }
+    
+                formulaLoopProtection[parentId] = true;
+    
+                // Convert range tokens
+                var tokensUpdate = function(tokens) {
+                    for (var index = 0; index < tokens.length; index++) {
+                        var f = [];
+                        var token = tokens[index].split(':');
+                        var e1 = jexcel.getIdFromColumnName(token[0], true);
+                        var e2 = jexcel.getIdFromColumnName(token[1], true);
+    
+                        if (e1[0] <= e2[0]) {
+                            var x1 = e1[0];
+                            var x2 = e2[0];
+                        } else {
+                            var x1 = e2[0];
+                            var x2 = e1[0];
+                        }
+    
+                        if (e1[1] <= e2[1]) {
+                            var y1 = e1[1];
+                            var y2 = e2[1];
+                        } else {
+                            var y1 = e2[1];
+                            var y2 = e1[1];
+                        }
+    
+                        for (var j = y1; j <= y2; j++) {
+                            for (var i = x1; i <= x2; i++) {
+                                f.push(jexcel.getColumnNameFromId([i, j]));
+                            }
+                        }
+    
+                        expression = expression.replace(tokens[index], f.join(','));
+                    }
+                }
+    
+                var tokens = expression.match(/([A-Z]+[0-9]+)\:([A-Z]+[0-9]+)/g);
+                if (tokens && tokens.length) {
+                    tokensUpdate(tokens);
+                }
+    
+                // String
+                var evalstring = '';
+    
+                // Get tokens
+                var tokens = expression.match(/([A-Z]+[0-9]+)/g);
+    
+                // Direct self-reference protection
+                if (tokens && tokens.indexOf(parentId) > -1) {
+                    console.error('Self Reference detected');
+                    return '#ERROR';
+                } else {
+                    if (tokens) {
+                        for (var i = 0; i < tokens.length; i++) {
+                            // Keep chain
+                            if (! obj.formula[tokens[i]]) {
+                                obj.formula[tokens[i]] = [];
+                            }
+                            // Is already in the register
+                            if (obj.formula[tokens[i]].indexOf(parentId) < 0) {
+                                obj.formula[tokens[i]].push(parentId);
+                            }
+    
+                            // Do not calculate again
+                            if (eval('typeof(' + tokens[i] + ') == "undefined"')) {
+                                // Coords
+                                var position = jexcel.getIdFromColumnName(tokens[i], 1);
+                                // Get value
+                                if (typeof(obj.options.data[position[1]]) != 'undefined' && typeof(obj.options.data[position[1]][position[0]]) != 'undefined') {
+                                    var value = obj.options.data[position[1]][position[0]];
+                                } else {
+                                    var value = '';
+                                }
+                                // Get column data
+                                if ((''+value).substr(0,1) == '=') {
+                                    if (formulaResults[tokens[i]]) {
+                                        value = formulaResults[tokens[i]];
+                                    } else {
+                                        value = execute(value, position[0], position[1]);
+                                        formulaResults[tokens[i]] = value;
+                                    }
+                                }
+                                // Type!
+                                if ((''+value).trim() == '') {
+                                    // Null
+                                    evalstring += "var " + tokens[i] + " = null;";
+                                } else {
+                                    if (value == Number(value) && obj.options.autoCasting == true) {
+                                        // Number
+                                        evalstring += "var " + tokens[i] + " = " + Number(value) + ";";
+                                    } else {
+                                        // Trying any formatted number
+                                        var number = obj.parseNumber(value, position[0])
+                                        if (obj.options.autoCasting == true && number) {
+                                            // Render as number
+                                            evalstring += "var " + tokens[i] + " = " + number + ";";
+                                        } else {
+                                            // Render as string
+                                            evalstring += "var " + tokens[i] + " = '" + value + "';";
+                                        }
+                                    }
+                                }
+                            }
+                        }
+                    }
+    
+                    // Convert formula to javascript
+                    try {
+                        evalstring += "function COLUMN() { return parseInt(x) + 1; }; function ROW() { return parseInt(y) + 1; }; function CELL() { return parentId; }; function TABLE() { return obj; }; function VALUE(col, row) { return obj.records[row-1][col-1].innerHTML; }; function THISROWCELL(col) { var id = jexcel.getIdFromColumnName(col+(parseInt(y)+1), true); return obj.records[id[1]][id[0]].innerHTML; }";
+    
+                        var res = eval(evalstring + expression.substr(1));
+                    } catch (e) {
+                        var res = '#ERROR';
+                    }
+    
+                    return res;
+                }
+            }
+    
+            return execute(expression, x, y);
+        }
+    
+        /**
+         * Trying to extract a number from a string
+         */
+        obj.parseNumber = function(value, columnNumber) {
+            // Decimal point
+            var decimal = columnNumber && obj.options.columns[columnNumber].decimal ? obj.options.columns[columnNumber].decimal : '.';
+    
+            // Parse both parts of the number
+            var number = ('' + value);
+            number = number.split(decimal);
+            number[0] = number[0].match(/[+-]?[0-9]/g);
+            if (number[0]) {
+                number[0] = number[0].join('');
+            }
+            if (number[1]) {
+                number[1] = number[1].match(/[0-9]*/g).join('');
+            }
+    
+            // Is a valid number
+            if (number[0] && Number(number[0]) >= 0) {
+                if (! number[1]) {
+                    var value = Number(number[0] + '.00');
+                } else {
+                    var value = Number(number[0] + '.' + number[1]);
+                }
+            } else {
+                var value = null;
+            }
+    
+            return value;
+        }
+    
+        /**
+         * Get row number
+         */
+        obj.row = function(cell) {
+        }
+    
+        /**
+         * Get col number
+         */
+        obj.col = function(cell) {
+        }
+    
+        obj.up = function(shiftKey, ctrlKey) {
+            if (shiftKey) {
+                if (obj.selectedCell[3] > 0) {
+                    obj.up.visible(1, ctrlKey ? 0 : 1)
+                }
+            } else {
+                if (obj.selectedCell[1] > 0) {
+                    obj.up.visible(0, ctrlKey ? 0 : 1)
+                }
+                obj.selectedCell[2] = obj.selectedCell[0];
+                obj.selectedCell[3] = obj.selectedCell[1];
+            }
+    
+            // Update selection
+            obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+    
+            // Change page
+            if (obj.options.lazyLoading == true) {
+                if (obj.selectedCell[1] == 0 || obj.selectedCell[3] == 0) {
+                    obj.loadPage(0);
+                    obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+                } else {
+                    if (obj.loadValidation()) {
+                        obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+                    } else {
+                        var item = parseInt(obj.tbody.firstChild.getAttribute('data-y'));
+                        if (obj.selectedCell[1] - item < 30) {
+                            obj.loadUp();
+                            obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+                        }
+                    }
+                }
+            } else if (obj.options.pagination > 0) {
+                var pageNumber = obj.whichPage(obj.selectedCell[3]);
+                if (pageNumber != obj.pageNumber) {
+                    obj.page(pageNumber);
+                }
+            }
+    
+            obj.updateScroll(1);
+        }
+    
+        obj.up.visible = function(group, direction) {
+            if (group == 0) {
+                var x = parseInt(obj.selectedCell[0]);
+                var y = parseInt(obj.selectedCell[1]);
+            } else {
+                var x = parseInt(obj.selectedCell[2]);
+                var y = parseInt(obj.selectedCell[3]);
+            }
+    
+            if (direction == 0) {
+                for (var j = 0; j < y; j++) {
+                    if (obj.records[j][x].style.display != 'none' && obj.rows[j].style.display != 'none') {
+                        y = j;
+                        break;
+                    }
+                }
+            } else {
+                y = obj.up.get(x, y);
+            }
+    
+            if (group == 0) {
+                obj.selectedCell[0] = x;
+                obj.selectedCell[1] = y;
+            } else {
+                obj.selectedCell[2] = x;
+                obj.selectedCell[3] = y;
+            }
+        }
+    
+        obj.up.get = function(x, y) {
+            var x = parseInt(x);
+            var y = parseInt(y);
+            for (var j = (y - 1); j >= 0; j--) {
+                if (obj.records[j][x].style.display != 'none' && obj.rows[j].style.display != 'none') {
+                    if (obj.records[j][x].getAttribute('data-merged')) {
+                        if (obj.records[j][x] == obj.records[y][x]) {
+                            continue;
+                        }
+                    }
+                    y = j;
+                    break;
+                }
+            }
+    
+            return y;
+        }
+    
+        obj.down = function(shiftKey, ctrlKey) {
+            if (shiftKey) {
+                if (obj.selectedCell[3] < obj.records.length - 1) {
+                    obj.down.visible(1, ctrlKey ? 0 : 1)
+                }
+            } else {
+                if (obj.selectedCell[1] < obj.records.length - 1) {
+                    obj.down.visible(0, ctrlKey ? 0 : 1)
+                }
+                obj.selectedCell[2] = obj.selectedCell[0];
+                obj.selectedCell[3] = obj.selectedCell[1];
+            }
+    
+            obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+    
+            // Change page
+            if (obj.options.lazyLoading == true) {
+                if ((obj.selectedCell[1] == obj.records.length - 1 || obj.selectedCell[3] == obj.records.length - 1)) {
+                    obj.loadPage(-1);
+                    obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+                } else {
+                    if (obj.loadValidation()) {
+                        obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+                    } else {
+                        var item = parseInt(obj.tbody.lastChild.getAttribute('data-y'));
+                        if (item - obj.selectedCell[3] < 30) {
+                            obj.loadDown();
+                            obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+                        }
+                    }
+                }
+            } else if (obj.options.pagination > 0) {
+                var pageNumber = obj.whichPage(obj.selectedCell[3]);
+                if (pageNumber != obj.pageNumber) {
+                    obj.page(pageNumber);
+                }
+            }
+    
+            obj.updateScroll(3);
+        }
+    
+        obj.down.visible = function(group, direction) {
+            if (group == 0) {
+                var x = parseInt(obj.selectedCell[0]);
+                var y = parseInt(obj.selectedCell[1]);
+            } else {
+                var x = parseInt(obj.selectedCell[2]);
+                var y = parseInt(obj.selectedCell[3]);
+            }
+    
+            if (direction == 0) {
+                for (var j = obj.rows.length - 1; j > y; j--) {
+                    if (obj.records[j][x].style.display != 'none' && obj.rows[j].style.display != 'none') {
+                        y = j;
+                        break;
+                    }
+                }
+            } else {
+                y = obj.down.get(x, y);
+            }
+    
+            if (group == 0) {
+                obj.selectedCell[0] = x;
+                obj.selectedCell[1] = y;
+            } else {
+                obj.selectedCell[2] = x;
+                obj.selectedCell[3] = y;
+            }
+        }
+    
+        obj.down.get = function(x, y) {
+            var x = parseInt(x);
+            var y = parseInt(y);
+            for (var j = (y + 1); j < obj.rows.length; j++) {
+                if (obj.records[j][x].style.display != 'none' && obj.rows[j].style.display != 'none') {
+                    if (obj.records[j][x].getAttribute('data-merged')) {
+                        if (obj.records[j][x] == obj.records[y][x]) {
+                            continue;
+                        }
+                    }
+                    y = j;
+                    break;
+                }
+            }
+    
+            return y;
+        }
+    
+        obj.right = function(shiftKey, ctrlKey) {
+            if (shiftKey) {
+                if (obj.selectedCell[2] < obj.headers.length - 1) {
+                    obj.right.visible(1, ctrlKey ? 0 : 1)
+                }
+            } else {
+                if (obj.selectedCell[0] < obj.headers.length - 1) {
+                    obj.right.visible(0, ctrlKey ? 0 : 1)
+                }
+                obj.selectedCell[2] = obj.selectedCell[0];
+                obj.selectedCell[3] = obj.selectedCell[1];
+            }
+    
+            obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+            obj.updateScroll(2);
+        }
+    
+        obj.right.visible = function(group, direction) {
+            if (group == 0) {
+                var x = parseInt(obj.selectedCell[0]);
+                var y = parseInt(obj.selectedCell[1]);
+            } else {
+                var x = parseInt(obj.selectedCell[2]);
+                var y = parseInt(obj.selectedCell[3]);
+            }
+    
+            if (direction == 0) {
+                for (var i = obj.headers.length - 1; i > x; i--) {
+                    if (obj.records[y][i].style.display != 'none') {
+                        x = i;
+                        break;
+                    }
+                }
+            } else {
+                x = obj.right.get(x, y);
+            }
+    
+            if (group == 0) {
+                obj.selectedCell[0] = x;
+                obj.selectedCell[1] = y;
+            } else {
+                obj.selectedCell[2] = x;
+                obj.selectedCell[3] = y;
+            }
+        }
+    
+        obj.right.get = function(x, y) {
+            var x = parseInt(x);
+            var y = parseInt(y);
+    
+            for (var i = (x + 1); i < obj.headers.length; i++) {
+                if (obj.records[y][i].style.display != 'none') {
+                    if (obj.records[y][i].getAttribute('data-merged')) {
+                        if (obj.records[y][i] == obj.records[y][x]) {
+                            continue;
+                        }
+                    }
+                    x = i;
+                    break;
+                }
+            }
+    
+            return x;
+        }
+    
+        obj.left = function(shiftKey, ctrlKey) {
+            if (shiftKey) {
+                if (obj.selectedCell[2] > 0) {
+                    obj.left.visible(1, ctrlKey ? 0 : 1)
+                }
+            } else {
+                if (obj.selectedCell[0] > 0) {
+                    obj.left.visible(0, ctrlKey ? 0 : 1)
+                }
+                obj.selectedCell[2] = obj.selectedCell[0];
+                obj.selectedCell[3] = obj.selectedCell[1];
+            }
+    
+            obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+            obj.updateScroll(0);
+        }
+    
+        obj.left.visible = function(group, direction) {
+            if (group == 0) {
+                var x = parseInt(obj.selectedCell[0]);
+                var y = parseInt(obj.selectedCell[1]);
+            } else {
+                var x = parseInt(obj.selectedCell[2]);
+                var y = parseInt(obj.selectedCell[3]);
+            }
+    
+            if (direction == 0) {
+                for (var i = 0; i < x; i++) {
+                    if (obj.records[y][i].style.display != 'none') {
+                        x = i;
+                        break;
+                    }
+                }
+            } else {
+                x = obj.left.get(x, y);
+            }
+    
+            if (group == 0) {
+                obj.selectedCell[0] = x;
+                obj.selectedCell[1] = y;
+            } else {
+                obj.selectedCell[2] = x;
+                obj.selectedCell[3] = y;
+            }
+        }
+    
+        obj.left.get = function(x, y) {
+            var x = parseInt(x);
+            var y = parseInt(y);
+            for (var i = (x - 1); i >= 0; i--) {
+                if (obj.records[y][i].style.display != 'none') {
+                    if (obj.records[y][i].getAttribute('data-merged')) {
+                        if (obj.records[y][i] == obj.records[y][x]) {
+                            continue;
+                        }
+                    }
+                    x = i;
+                    break;
+                }
+            }
+    
+            return x;
+        }
+    
+        obj.first = function(shiftKey, ctrlKey) {
+            if (shiftKey) {
+                if (ctrlKey) {
+                    obj.selectedCell[3] = 0;
+                } else {
+                    obj.left.visible(1, 0);
+                }
+            } else {
+                if (ctrlKey) {
+                    obj.selectedCell[1] = 0;
+                } else {
+                    obj.left.visible(0, 0);
+                }
+                obj.selectedCell[2] = obj.selectedCell[0];
+                obj.selectedCell[3] = obj.selectedCell[1];
+            }
+    
+            // Change page
+            if (obj.options.lazyLoading == true && (obj.selectedCell[1] == 0 || obj.selectedCell[3] == 0)) {
+                obj.loadPage(0);
+            } else if (obj.options.pagination > 0) {
+                var pageNumber = obj.whichPage(obj.selectedCell[3]);
+                if (pageNumber != obj.pageNumber) {
+                    obj.page(pageNumber);
+                }
+            }
+    
+            obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+            obj.updateScroll(1);
+        }
+    
+        obj.last = function(shiftKey, ctrlKey) {
+            if (shiftKey) {
+                if (ctrlKey) {
+                    obj.selectedCell[3] = obj.records.length - 1;
+                } else {
+                    obj.right.visible(1, 0);
+                }
+            } else {
+                if (ctrlKey) {
+                    obj.selectedCell[1] = obj.records.length - 1;
+                } else {
+                    obj.right.visible(0, 0);
+                }
+                obj.selectedCell[2] = obj.selectedCell[0];
+                obj.selectedCell[3] = obj.selectedCell[1];
+            }
+    
+            // Change page
+            if (obj.options.lazyLoading == true && (obj.selectedCell[1] == obj.records.length - 1 || obj.selectedCell[3] == obj.records.length - 1)) {
+                obj.loadPage(-1);
+            } else if (obj.options.pagination > 0) {
+                var pageNumber = obj.whichPage(obj.selectedCell[3]);
+                if (pageNumber != obj.pageNumber) {
+                    obj.page(pageNumber);
+                }
+            }
+    
+            obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+            obj.updateScroll(3);
+        }
+    
+        obj.selectAll = function() {
+            if (! obj.selectedCell) {
+                obj.selectedCell = [];
+            }
+    
+            obj.selectedCell[0] = 0;
+            obj.selectedCell[1] = 0;
+            obj.selectedCell[2] = obj.headers.length - 1;
+            obj.selectedCell[3] = obj.records.length - 1;
+    
+            obj.updateSelectionFromCoords(obj.selectedCell[0], obj.selectedCell[1], obj.selectedCell[2], obj.selectedCell[3]);
+        }
+    
+        /**
+         * Go to a page in a lazyLoading
+         */
+        obj.loadPage = function(pageNumber) {
+            // Search
+            if (obj.options.search == true && obj.results) {
+                var results = obj.results;
+            } else {
+                var results = obj.rows;
+            }
+    
+            // Per page
+            var quantityPerPage = 100;
+    
+            // pageNumber
+            if (pageNumber == null || pageNumber == -1) {
+                // Last page
+                pageNumber = Math.ceil(results.length / quantityPerPage); 
+            }
+    
+            var startRow = (pageNumber * quantityPerPage);
+            var finalRow = (pageNumber * quantityPerPage) + quantityPerPage;
+            if (finalRow > results.length) {
+                finalRow = results.length;
+            }
+            startRow = finalRow - 100;
+            if (startRow < 0) {
+                startRow = 0;
+            }
+    
+            // Appeding items
+            for (var j = startRow; j < finalRow; j++) {
+                if (obj.options.search == true && obj.results) {
+                    obj.tbody.appendChild(obj.rows[results[j]]);
+                } else {
+                    obj.tbody.appendChild(obj.rows[j]);
+                }
+    
+                if (obj.tbody.children.length > quantityPerPage) {
+                    obj.tbody.removeChild(obj.tbody.firstChild);
+                }
+            }
+        }
+    
+        obj.loadUp = function() {
+            // Search
+            if (obj.options.search == true && obj.results) {
+                var results = obj.results;
+            } else {
+                var results = obj.rows;
+            }
+            var test = 0;
+            if (results.length > 100) {
+                // Get the first element in the page
+                var item = parseInt(obj.tbody.firstChild.getAttribute('data-y'));
+                if (obj.options.search == true && obj.results) {
+                    item = results.indexOf(item);
+                }
+                if (item > 0) {
+                    for (var j = 0; j < 30; j++) {
+                        item = item - 1;
+                        if (item > -1) {
+                            if (obj.options.search == true && obj.results) {
+                                obj.tbody.insertBefore(obj.rows[results[item]], obj.tbody.firstChild);
+                            } else {
+                                obj.tbody.insertBefore(obj.rows[item], obj.tbody.firstChild);
+                            }
+                            if (obj.tbody.children.length > 100) {
+                                obj.tbody.removeChild(obj.tbody.lastChild);
+                                test = 1;
+                            }
+                        }
+                    }
+                }
+            }
+            return test;
+        }
+    
+        obj.loadDown = function() {
+            // Search
+            if (obj.options.search == true && obj.results) {
+                var results = obj.results;
+            } else {
+                var results = obj.rows;
+            }
+            var test = 0;
+            if (results.length > 100) {
+                // Get the last element in the page
+                var item = parseInt(obj.tbody.lastChild.getAttribute('data-y'));
+                if (obj.options.search == true && obj.results) {
+                    item = results.indexOf(item);
+                }
+                if (item < obj.rows.length - 1) {
+                    for (var j = 0; j <= 30; j++) {
+                        if (item < results.length) {
+                            if (obj.options.search == true && obj.results) {
+                                obj.tbody.appendChild(obj.rows[results[item]]);
+                            } else {
+                                obj.tbody.appendChild(obj.rows[item]);
+                            }
+                            if (obj.tbody.children.length > 100) {
+                                obj.tbody.removeChild(obj.tbody.firstChild);
+                                test = 1;
+                            }
+                        }
+                        item = item + 1;
+                    }
+                }
+            }
+    
+            return test;
+        }
+    
+        obj.loadValidation = function() {
+            if (obj.selectedCell) {
+                var currentPage = parseInt(obj.tbody.firstChild.getAttribute('data-y')) / 100;
+                var selectedPage = parseInt(obj.selectedCell[3] / 100);
+                var totalPages = parseInt(obj.rows.length / 100);
+    
+                if (currentPage != selectedPage && selectedPage <= totalPages) {
+                    if (! Array.prototype.indexOf.call(obj.tbody.children, obj.rows[obj.selectedCell[3]])) {
+                        obj.loadPage(selectedPage);
+                        return true;
+                    }
+                }
+            }
+    
+            return false;
+        }
+    
+        /**
+         * Reset search
+         */
+        obj.resetSearch = function() {
+            obj.searchInput.value = '';
+            obj.search('');
+            obj.results = null;
+        }
+
+        /**
+         * Search
+         */
+        obj.search = function(query) {
+            // Query
+            if (query) {
+                var query = query.toLowerCase();
+            }
+    
+            // Reset any filter
+            if (obj.options.filters) {
+                obj.resetFilters();
+            }
+
+            // Reset selection
+            obj.resetSelection();
+    
+            // Total of results
+            obj.pageNumber = 0;
+            obj.results = [];
+    
+            if (query) {
+                // Search filter
+                var search = function(item, query, index) {
+                    for (var i = 0; i < item.length; i++) {
+                        if ((''+item[i]).toLowerCase().search(query) >= 0 ||
+                            (''+obj.records[index][i].innerHTML).toLowerCase().search(query) >= 0) {
+                            return true;
+                        }
+                    }
+                    return false;
+                }
+    
+                // Result
+                var addToResult = function(k) {
+                    if (obj.results.indexOf(k) == -1) {
+                        obj.results.push(k);
+                    }
+                }
+    
+                // Filter
+                var data = obj.options.data.filter(function(v, k) {
+                    if (search(v, query, k)) {
+                        // Merged rows found
+                        var rows = obj.isRowMerged(k);
+                        if (rows.length) {
+                            for (var i = 0; i < rows.length; i++) {
+                                var row = jexcel.getIdFromColumnName(rows[i], true);
+                                for (var j = 0; j < obj.options.mergeCells[rows[i]][1]; j++) {
+                                    addToResult(row[1]+j);
+                                }
+                            }
+                        } else {
+                            // Normal row found
+                            addToResult(k);
+                        }
+                        return true;
+                    } else {
+                        return false;
+                    }
+                });
+            } else {
+                obj.results = null;
+            }
+
+            return obj.updateResult();
+        }
+    
+        obj.updateResult = function() {
+            var total = 0;
+            var index = 0;
+    
+            // Page 1
+            if (obj.options.lazyLoading == true) {
+                total = 100;
+            } else if (obj.options.pagination > 0) {
+                total = obj.options.pagination;
+            } else {
+                if (obj.results) {
+                    total = obj.results.length;
+                } else {
+                    total = obj.rows.length;
+                }
+            }
+    
+            // Reset current nodes
+            while (obj.tbody.firstChild) {
+                obj.tbody.removeChild(obj.tbody.firstChild);
+            }
+    
+            // Hide all records from the table
+            for (var j = 0; j < obj.rows.length; j++) {
+                if (! obj.results || obj.results.indexOf(j) > -1) {
+                    if (index < total) {
+                        obj.tbody.appendChild(obj.rows[j]);
+                        index++;
+                    }
+                    obj.rows[j].style.display = '';
+                } else {
+                    obj.rows[j].style.display = 'none';
+                }
+            }
+    
+            // Update pagination
+            if (obj.options.pagination > 0) {
+                obj.updatePagination();
+            }
+
+            obj.updateCornerPosition();
+    
+            return total;
+        }
+
+        /**
+         * Which page the cell is
+         */
+        obj.whichPage = function(cell) {
+            // Search
+            if (obj.options.search == true && obj.results) {
+                cell = obj.results.indexOf(cell);
+            }
+    
+            return (Math.ceil((parseInt(cell) + 1) / parseInt(obj.options.pagination))) - 1;
+        }
+    
+        /**
+         * Go to page
+         */
+        obj.page = function(pageNumber) {
+            var oldPage = obj.pageNumber;
+
+            // Search
+            if (obj.options.search == true && obj.results) {
+                var results = obj.results;
+            } else {
+                var results = obj.rows;
+            }
+    
+            // Per page
+            var quantityPerPage = parseInt(obj.options.pagination);
+    
+            // pageNumber
+            if (pageNumber == null || pageNumber == -1) {
+                // Last page
+                pageNumber = Math.ceil(results.length / quantityPerPage); 
+            }
+    
+            // Page number
+            obj.pageNumber = pageNumber;
+    
+            var startRow = (pageNumber * quantityPerPage);
+            var finalRow = (pageNumber * quantityPerPage) + quantityPerPage;
+            if (finalRow > results.length) {
+                finalRow = results.length;
+            }
+            if (startRow < 0) {
+                startRow = 0;
+            }
+    
+            // Reset container
+            while (obj.tbody.firstChild) {
+                obj.tbody.removeChild(obj.tbody.firstChild);
+            }
+    
+            // Appeding items
+            for (var j = startRow; j < finalRow; j++) {
+                if (obj.options.search == true && obj.results) {
+                    obj.tbody.appendChild(obj.rows[results[j]]);
+                } else {
+                    obj.tbody.appendChild(obj.rows[j]);
+                }
+            }
+    
+            if (obj.options.pagination > 0) {
+                obj.updatePagination();
+            }
+    
+            // Update corner position
+            obj.updateCornerPosition();
+
+            // Events
+            obj.dispatch('onchangepage', el, pageNumber, oldPage);
+        }
+    
+        /**
+         * Update the pagination
+         */
+        obj.updatePagination = function() {
+            // Reset container
+            obj.pagination.children[0].innerHTML = '';
+            obj.pagination.children[1].innerHTML = '';
+    
+            // Start pagination
+            if (obj.options.pagination) {
+                // Searchable
+                if (obj.options.search == true && obj.results) {
+                    var results = obj.results.length;
+                } else {
+                    var results = obj.rows.length;
+                }
+    
+                if (! results) {
+                    // No records found
+                    obj.pagination.children[0].innerHTML = obj.options.text.noRecordsFound;
+                } else {
+                    // Pagination container
+                    var quantyOfPages = Math.ceil(results / obj.options.pagination);
+    
+                    if (obj.pageNumber < 6) {
+                        var startNumber = 1;
+                        var finalNumber = quantyOfPages < 10 ? quantyOfPages : 10;
+                    } else if (quantyOfPages - obj.pageNumber < 5) {
+                        var startNumber = quantyOfPages - 9;
+                        var finalNumber = quantyOfPages;
+                        if (startNumber < 1) {
+                            startNumber = 1;
+                        }
+                    } else {
+                        var startNumber = obj.pageNumber - 4;
+                        var finalNumber = obj.pageNumber + 5;
+                    }
+    
+                    // First
+                    if (startNumber > 1) {
+                        var paginationItem = document.createElement('div');
+                        paginationItem.className = 'jexcel_page';
+                        paginationItem.innerHTML = '<';
+                        paginationItem.title = 1;
+                        obj.pagination.children[1].appendChild(paginationItem);
+                    }
+    
+                    // Get page links
+                    for (var i = startNumber; i <= finalNumber; i++) {
+                        var paginationItem = document.createElement('div');
+                        paginationItem.className = 'jexcel_page';
+                        paginationItem.innerHTML = i;
+                        obj.pagination.children[1].appendChild(paginationItem);
+    
+                        if (obj.pageNumber == (i-1)) {
+                            paginationItem.classList.add('jexcel_page_selected');
+                        }
+                    }
+    
+                    // Last
+                    if (finalNumber < quantyOfPages) {
+                        var paginationItem = document.createElement('div');
+                        paginationItem.className = 'jexcel_page';
+                        paginationItem.innerHTML = '>';
+                        paginationItem.title = quantyOfPages;
+                        obj.pagination.children[1].appendChild(paginationItem);
+                    }
+    
+                    // Text
+                    var format = function(format) {
+                        var args = Array.prototype.slice.call(arguments, 1);
+                        return format.replace(/{(\d+)}/g, function(match, number) {
+                          return typeof args[number] != 'undefined'
+                            ? args[number]
+                            : match
+                          ;
+                        });
+                    };
+    
+                    obj.pagination.children[0].innerHTML = format(obj.options.text.showingPage, obj.pageNumber + 1, quantyOfPages)
+                }
+            }
+        }
+    
+        /**
+         * Download CSV table
+         * 
+         * @return null
+         */
+        obj.download = function(includeHeaders) {
+            if (obj.options.allowExport == false) {
+                console.error('Export not allowed');
+            } else {
+                // Data
+                var data = '';
+                if (includeHeaders == true || obj.options.includeHeadersOnDownload == true) {
+                    data += obj.getHeaders();
+                    data += "\r\n";
+                }
+
+                // Get data
+                data += obj.copy(false, obj.options.csvDelimiter, true);
+
+                // Download element
+                var blob = new Blob(["\uFEFF"+data], {type: 'text/csv;charset=utf-8;'});
+
+                // IE Compatibility
+                if (window.navigator && window.navigator.msSaveOrOpenBlob) {
+                    window.navigator.msSaveOrOpenBlob(blob, options.csvFileName + '.csv');
+                } else {
+                    // Download element
+                    var pom = document.createElement('a');
+                    var url = URL.createObjectURL(blob);
+                    pom.href = url;
+                    pom.setAttribute('download', obj.options.csvFileName + '.csv');
+                    document.body.appendChild(pom);
+                    pom.click();
+                    pom.parentNode.removeChild(pom);
+                }
+            }
+        }
+    
+        /**
+         * Initializes a new history record for undo/redo
+         * 
+         * @return null
+         */
+        obj.setHistory = function(changes) {
+            if (obj.ignoreHistory != true) {
+                // Increment and get the current history index
+                var index = ++obj.historyIndex;
+    
+                // Slice the array to discard undone changes
+                obj.history = (obj.history = obj.history.slice(0, index + 1));
+    
+                // Keep history
+                obj.history[index] = changes;
+            }
+        }
+    
+        /**
+         * Copy method
+         * 
+         * @param bool highlighted - Get only highlighted cells
+         * @param delimiter - \t default to keep compatibility with excel
+         * @return string value
+         */
+        obj.copy = function(highlighted, delimiter, returnData) {
+            if (! delimiter) {
+                delimiter = "\t";
+            }
+    
+            // Controls
+            var col = [];
+            var colLabel = [];
+            var row = [];
+            var rowLabel = [];
+            var x = obj.options.data[0].length
+            var y = obj.options.data.length
+            var tmp = '';
+
+            // Headers
+            if (obj.options.includeHeadersOnCopy == true) {
+                if (obj.options.copyCompatibility == true) {
+                    strLabel.push(obj.getHeaders(true).join(delimiter));
+                } else {
+                    str.push(obj.getHeaders(true).join(delimiter));
+                }
+            }
+
+            // Reset container
+            obj.style = [];
+    
+            // Go through the columns to get the data
+            for (var j = 0; j < y; j++) {
+                col = [];
+                colLabel = [];
+    
+                for (var i = 0; i < x; i++) {
+                    // If cell is highlighted
+                    if (! highlighted || obj.records[j][i].classList.contains('highlight')) {
+                        // Values
+                        var value = obj.options.data[j][i];
+                        if (value.match && (value.match(/,/g) || value.match(/\n/) || value.match(/\"/))) {
+                            value = value.replace(new RegExp('"', 'g'), '""');
+                            value = '"' + value + '"';
+                        }
+                        col.push(value);
+    
+                        // Labels
+                        if (obj.options.columns[i].type == 'checkbox' || obj.options.columns[i].type == 'radio') {
+                            var label = value;
+                        } else {
+                            var label = obj.records[j][i].innerHTML;
+                            if (label.match && (label.match(/,/g) || label.match(/\n/) || label.match(/\"/))) {
+                                // Scape double quotes
+                                label = label.replace(new RegExp('"', 'g'), '""');
+                                label = '"' + label + '"';
+                            }
+                        }
+                        colLabel.push(label);
+    
+                        // Get style
+                        tmp = obj.records[j][i].getAttribute('style');
+                        tmp = tmp.replace('display: none;', '');
+                        obj.style.push(tmp ? tmp : '');
+                    }
+                }
+    
+                if (col.length) {
+                    row.push(col.join(delimiter));
+                }
+                if (colLabel.length) {
+                    rowLabel.push(colLabel.join(delimiter));
+                }
+            }
+
+            // Final string
+            var str = row.join("\r\n");
+            var strLabel = rowLabel.join("\r\n");
+
+            // Create a hidden textarea to copy the values
+            if (! returnData) {
+                if (obj.options.copyCompatibility == true) {
+                    obj.textarea.value = strLabel;
+                } else {
+                    obj.textarea.value = str;
+                }
+                obj.textarea.select();
+                document.execCommand("copy");
+            }
+    
+            // Keep data
+            if (obj.options.copyCompatibility == true) {
+                obj.data = strLabel;
+            } else {
+                obj.data = str;
+            }
+            // Keep non visible information
+            obj.hashString = obj.hash(obj.data);
+    
+            // Any exiting border should go
+            obj.removeCopyingSelection();
+
+            // Border
+            if (obj.highlighted) {
+                for (var i = 0; i < obj.highlighted.length; i++) {
+                    obj.highlighted[i].classList.add('copying');
+                    if (obj.highlighted[i].classList.contains('highlight-left')) {
+                        obj.highlighted[i].classList.add('copying-left');
+                    }
+                    if (obj.highlighted[i].classList.contains('highlight-right')) {
+                        obj.highlighted[i].classList.add('copying-right');
+                    }
+                    if (obj.highlighted[i].classList.contains('highlight-top')) {
+                        obj.highlighted[i].classList.add('copying-top');
+                    }
+                    if (obj.highlighted[i].classList.contains('highlight-bottom')) {
+                        obj.highlighted[i].classList.add('copying-bottom');
+                    }
+                }
+            }
+
+            // Paste event
+            obj.dispatch('oncopy', el, obj.options.copyCompatibility == true ? rowLabel : row, obj.hashString);
+
+            return obj.data;
+        }
+    
+        /**
+         * jExcel paste method
+         * 
+         * @param integer row number
+         * @return string value
+         */
+        obj.paste = function(x, y, data) {
+            // Paste filter
+            var ret = obj.dispatch('onbeforepaste', el, data, x, y);
+
+            if (ret === false) {
+                return false;
+            } else if (ret) {
+                var data = ret;
+            }
+    
+            // Controls
+            var hash = obj.hash(data);
+            var style = (hash == obj.hashString) ? obj.style : null;
+    
+            // Depending on the behavior
+            if (obj.options.copyCompatibility == true && hash == obj.hashString) {
+                var data = obj.data;
+            }
+    
+            // Split new line
+            var data = obj.parseCSV(data, "\t");
+    
+            if (x != null && y != null && data) {
+                // Records
+                var i = 0;
+                var j = 0;
+                var records = [];
+                var newStyle = {};
+                var oldStyle = {};
+                var styleIndex = 0;
+    
+                // Index
+                var colIndex = parseInt(x);
+                var rowIndex = parseInt(y);
+                var row = null;
+    
+                // Go through the columns to get the data
+                while (row = data[j]) {
+                    i = 0;
+                    colIndex = parseInt(x);
+    
+                    while (row[i] != null) {
+                        // Update and keep history
+                        var record = obj.updateCell(colIndex, rowIndex, row[i]);
+                        // Keep history
+                        records.push(record);
+                        // Update all formulas in the chain
+                        obj.updateFormulaChain(colIndex, rowIndex, records);
+                        // Style
+                        if (style && style[styleIndex]) {
+                            var columnName = jexcel.getColumnNameFromId([colIndex, rowIndex]);
+                            newStyle[columnName] = style[styleIndex];
+                            oldStyle[columnName] = obj.getStyle(columnName);
+                            obj.records[rowIndex][colIndex].setAttribute('style', style[styleIndex]);
+                            styleIndex++
+                        }
+                        i++;
+                        if (row[i] != null) {
+                            if (colIndex >= obj.headers.length - 1) {
+                                obj.insertColumn();
+                            }
+                            colIndex = obj.right.get(colIndex, rowIndex);
+                        }
+                    }
+    
+                    j++;
+                    if (data[j]) {
+                        if (rowIndex >= obj.rows.length-1) {
+                            obj.insertRow();
+                        }
+                        rowIndex = obj.down.get(x, rowIndex);
+                    }
+                }
+    
+                // Select the new cells
+                obj.updateSelectionFromCoords(x, y, colIndex, rowIndex);
+    
+                // Update history
+                obj.setHistory({
+                    action:'setValue',
+                    records:records,
+                    selection:obj.selectedCell,
+                    newStyle:newStyle,
+                    oldStyle:oldStyle,
+                });
+    
+                // Update table
+                obj.updateTable();
+    
+                // Paste event
+                obj.dispatch('onpaste', el, data);
+    
+                // On after changes
+                obj.onafterchanges(el, records);
+            }
+
+            obj.removeCopyingSelection();
+        }
+
+        /**
+         * Remove copying border
+         */
+        obj.removeCopyingSelection = function() {
+            var copying = document.querySelectorAll('.jexcel .copying');
+            for (var i = 0; i < copying.length; i++) {
+                copying[i].classList.remove('copying');
+                copying[i].classList.remove('copying-left');
+                copying[i].classList.remove('copying-right');
+                copying[i].classList.remove('copying-top');
+                copying[i].classList.remove('copying-bottom');
+            }
+        }
+
+        /**
+         * Process row
+         */
+        obj.historyProcessRow = function(type, historyRecord) {
+            var rowIndex = (! historyRecord.insertBefore) ? historyRecord.rowNumber + 1 : historyRecord.rowNumber;
+    
+            if (obj.options.search == true) {
+                if (obj.results && obj.results.length != obj.rows.length) {
+                    obj.resetSearch();
+                }
+            }
+    
+            // Remove row
+            if (type == 1) {
+                var numOfRows = historyRecord.numOfRows;
+                // Remove nodes
+                for (var j = rowIndex; j < (numOfRows + rowIndex); j++) {
+                    obj.rows[j].parentNode.removeChild(obj.rows[j]);
+                }
+                // Remove references
+                obj.records.splice(rowIndex, numOfRows);
+                obj.options.data.splice(rowIndex, numOfRows);
+                obj.rows.splice(rowIndex, numOfRows);
+    
+                obj.conditionalSelectionUpdate(1, rowIndex, (numOfRows + rowIndex) - 1);
+            } else {
+                // Insert data
+                obj.records = jexcel.injectArray(obj.records, rowIndex, historyRecord.rowRecords);
+                obj.options.data = jexcel.injectArray(obj.options.data, rowIndex, historyRecord.rowData);
+                obj.rows = jexcel.injectArray(obj.rows, rowIndex, historyRecord.rowNode);
+                // Insert nodes
+                var index = 0
+                for (var j = rowIndex; j < (historyRecord.numOfRows + rowIndex); j++) {
+                    obj.tbody.insertBefore(historyRecord.rowNode[index], obj.tbody.children[j]);
+                    index++;
+                }
+            }
+    
+            // Respect pagination
+            if (obj.options.pagination > 0) {
+                obj.page(obj.pageNumber);
+            }
+    
+            obj.updateTableReferences();
+        }
+    
+        /**
+         * Process column
+         */
+        obj.historyProcessColumn = function(type, historyRecord) {
+            var columnIndex = (! historyRecord.insertBefore) ? historyRecord.columnNumber + 1 : historyRecord.columnNumber;
+    
+            // Remove column
+            if (type == 1) {
+                var numOfColumns = historyRecord.numOfColumns;
+    
+                obj.options.columns.splice(columnIndex, numOfColumns);
+                for (var i = columnIndex; i < (numOfColumns + columnIndex); i++) {
+                    obj.headers[i].parentNode.removeChild(obj.headers[i]);
+                    obj.colgroup[i].parentNode.removeChild(obj.colgroup[i]);
+                }
+                obj.headers.splice(columnIndex, numOfColumns);
+                obj.colgroup.splice(columnIndex, numOfColumns);
+                for (var j = 0; j < historyRecord.data.length; j++) {
+                    for (var i = columnIndex; i < (numOfColumns + columnIndex); i++) {
+                        obj.records[j][i].parentNode.removeChild(obj.records[j][i]);
+                    }
+                    obj.records[j].splice(columnIndex, numOfColumns);
+                    obj.options.data[j].splice(columnIndex, numOfColumns);
+                }
+                // Process footers
+                if (obj.options.footers) {
+                    for (var j = 0; j < obj.options.footers.length; j++) {
+                        obj.options.footers[j].splice(columnIndex, numOfColumns);
+                    }
+                }
+            } else {
+                // Insert data
+                obj.options.columns = jexcel.injectArray(obj.options.columns, columnIndex, historyRecord.columns);
+                obj.headers = jexcel.injectArray(obj.headers, columnIndex, historyRecord.headers);
+                obj.colgroup = jexcel.injectArray(obj.colgroup, columnIndex, historyRecord.colgroup);
+    
+                var index = 0
+                for (var i = columnIndex; i < (historyRecord.numOfColumns + columnIndex); i++) {
+                    obj.headerContainer.insertBefore(historyRecord.headers[index], obj.headerContainer.children[i+1]);
+                    obj.colgroupContainer.insertBefore(historyRecord.colgroup[index], obj.colgroupContainer.children[i+1]);
+                    index++;
+                }
+    
+                for (var j = 0; j < historyRecord.data.length; j++) {
+                    obj.options.data[j] = jexcel.injectArray(obj.options.data[j], columnIndex, historyRecord.data[j]);
+                    obj.records[j] = jexcel.injectArray(obj.records[j], columnIndex, historyRecord.records[j]);
+                    var index = 0
+                    for (var i = columnIndex; i < (historyRecord.numOfColumns + columnIndex); i++) {
+                        obj.rows[j].insertBefore(historyRecord.records[j][index], obj.rows[j].children[i+1]);
+                        index++;
+                    }
+                }
+                // Process footers
+                if (obj.options.footers) {
+                    for (var j = 0; j < obj.options.footers.length; j++) {
+                        obj.options.footers[j] = jexcel.injectArray(obj.options.footers[j], columnIndex, historyRecord.footers[j]);
+                    }
+                }
+            }
+    
+            // Adjust nested headers
+            if (obj.options.nestedHeaders && obj.options.nestedHeaders.length > 0) {
+                // Flexible way to handle nestedheaders
+                if (obj.options.nestedHeaders[0] && obj.options.nestedHeaders[0][0]) {
+                    for (var j = 0; j < obj.options.nestedHeaders.length; j++) {
+                        if (type == 1) {
+                            var colspan = parseInt(obj.options.nestedHeaders[j][obj.options.nestedHeaders[j].length-1].colspan) - historyRecord.numOfColumns;
+                        } else {
+                            var colspan = parseInt(obj.options.nestedHeaders[j][obj.options.nestedHeaders[j].length-1].colspan) + historyRecord.numOfColumns;
+                        }
+                        obj.options.nestedHeaders[j][obj.options.nestedHeaders[j].length-1].colspan = colspan;
+                        obj.thead.children[j].children[obj.thead.children[j].children.length-1].setAttribute('colspan', colspan);
+                    }
+                } else {
+                    if (type == 1) {
+                        var colspan = parseInt(obj.options.nestedHeaders[0].colspan) - historyRecord.numOfColumns;
+                    } else {
+                        var colspan = parseInt(obj.options.nestedHeaders[0].colspan) + historyRecord.numOfColumns;
+                    }
+                    obj.options.nestedHeaders[0].colspan = colspan;
+                    obj.thead.children[0].children[obj.thead.children[0].children.length-1].setAttribute('colspan', colspan);
+                }
+            }
+    
+            obj.updateTableReferences();
+        }
+    
+        /**
+         * Undo last action
+         */
+        obj.undo = function() {
+            // Ignore events and history
+            var ignoreEvents = obj.ignoreEvents ? true : false;
+            var ignoreHistory = obj.ignoreHistory ? true : false;
+    
+            obj.ignoreEvents = true;
+            obj.ignoreHistory = true;
+    
+            // Records
+            var records = [];
+    
+            // Update cells
+            if (obj.historyIndex >= 0) {
+                // History
+                var historyRecord = obj.history[obj.historyIndex--];
+    
+                if (historyRecord.action === 'endResizeTable') {
+                    while (obj.history[obj.historyIndex].action !== "beginResizeTable") {
+                    obj.undo();
+                  }
+                  obj.undo();
+                } else if (historyRecord.action == 'endChangeType') {
+                  obj.options.columns[historyRecord.column].type = historyRecord.oldType;
+                  while (obj.history[obj.historyIndex].action !== "beginChangeType") {
+                    obj.undo();
+                  }
+                  obj.undo();
+                } else if (historyRecord.action == 'insertRow') {
+                    obj.historyProcessRow(1, historyRecord);
+                } else if (historyRecord.action == 'deleteRow') {
+                    obj.historyProcessRow(0, historyRecord);
+                } else if (historyRecord.action == 'insertColumn') {
+                    obj.historyProcessColumn(1, historyRecord);
+                } else if (historyRecord.action == 'deleteColumn') {
+                    obj.historyProcessColumn(0, historyRecord);
+                } else if (historyRecord.action == 'moveRow') {
+                    obj.moveRow(historyRecord.newValue, historyRecord.oldValue);
+                } else if (historyRecord.action == 'moveColumn') {
+                    obj.moveColumn(historyRecord.newValue, historyRecord.oldValue);
+                } else if (historyRecord.action == 'setMerge') {
+                    obj.removeMerge(historyRecord.column, historyRecord.data);
+                } else if (historyRecord.action == 'setStyle') {
+                    obj.setStyle(historyRecord.oldValue, null, null, 1);
+                } else if (historyRecord.action == 'setWidth') {
+                    obj.setWidth(historyRecord.column, historyRecord.oldValue);
+                } else if (historyRecord.action == 'setHeight') {
+                    obj.setHeight(historyRecord.row, historyRecord.oldValue);
+                } else if (historyRecord.action == 'setHeader') {
+                    obj.setHeader(historyRecord.column, historyRecord.oldValue);
+                } else if (historyRecord.action == 'setComments') {
+                    obj.setComments(historyRecord.column, historyRecord.oldValue[0], historyRecord.oldValue[1]);
+                } else if (historyRecord.action == 'orderBy') {
+                    var rows = [];
+                    for (var j = 0; j < historyRecord.rows.length; j++) {
+                        rows[historyRecord.rows[j]] = j;
+                    }
+                    obj.updateOrderArrow(historyRecord.column, historyRecord.order ? 0 : 1);
+                    obj.updateOrder(rows);
+                } else if (historyRecord.action == 'setValue') {
+                    // Redo for changes in cells
+                    for (var i = 0; i < historyRecord.records.length; i++) {
+                        records.push({
+                            x: historyRecord.records[i].x,
+                            y: historyRecord.records[i].y,
+                            newValue: historyRecord.records[i].oldValue,
+                        });
+
+                        if (historyRecord.oldStyle) {
+                            obj.resetStyle(historyRecord.oldStyle);
+                        }
+                    }
+                    // Update records
+                    obj.setValue(records);
+
+                    // Update selection
+                    if (historyRecord.selection) {
+                        obj.updateSelectionFromCoords(historyRecord.selection[0], historyRecord.selection[1], historyRecord.selection[2], historyRecord.selection[3]);
+                    }
+                }
+            }
+            obj.ignoreEvents = ignoreEvents;
+            obj.ignoreHistory = ignoreHistory;
+    
+            // Events
+            obj.dispatch('onundo', el, historyRecord);
+        }
+    
+        /**
+         * Redo previously undone action
+         */
+        obj.redo = function() {
+            // Ignore events and history
+            var ignoreEvents = obj.ignoreEvents ? true : false;
+            var ignoreHistory = obj.ignoreHistory ? true : false;
+    
+            obj.ignoreEvents = true;
+            obj.ignoreHistory = true;
+    
+            // Records
+            var records = [];
+            
+            // Update cells
+            if (obj.historyIndex < obj.history.length - 1) {
+                // History
+                var historyRecord = obj.history[++obj.historyIndex];
+                if (historyRecord.action == 'beginResizeTable') {
+                    while (obj.history[obj.historyIndex].action !== "endResizeTable") {
+                    obj.redo();
+                  }
+                  obj.redo();
+                } else if (historyRecord.action == 'beginChangeType') {
+                  obj.options.columns[historyRecord.column].type = historyRecord.newType;
+                  while (obj.history[obj.historyIndex].action !== "endChangeType") {
+                    obj.redo();
+                  }
+                  obj.redo();
+                } else if (historyRecord.action == 'insertRow') {
+                    obj.historyProcessRow(0, historyRecord);
+                } else if (historyRecord.action == 'deleteRow') {
+                    obj.historyProcessRow(1, historyRecord);
+                } else if (historyRecord.action == 'insertColumn') {
+                    obj.historyProcessColumn(0, historyRecord);
+                } else if (historyRecord.action == 'deleteColumn') {
+                    obj.historyProcessColumn(1, historyRecord);
+                } else if (historyRecord.action == 'moveRow') {
+                    obj.moveRow(historyRecord.oldValue, historyRecord.newValue);
+                } else if (historyRecord.action == 'moveColumn') {
+                    obj.moveColumn(historyRecord.oldValue, historyRecord.newValue);
+                } else if (historyRecord.action == 'setMerge') {
+                    obj.setMerge(historyRecord.column, historyRecord.colspan, historyRecord.rowspan, 1);
+                } else if (historyRecord.action == 'setStyle') {
+                    obj.setStyle(historyRecord.newValue, null, null, 1);
+                } else if (historyRecord.action == 'setWidth') {
+                    obj.setWidth(historyRecord.column, historyRecord.newValue);
+                } else if (historyRecord.action == 'setHeight') {
+                    obj.setHeight(historyRecord.row, historyRecord.newValue);
+                } else if (historyRecord.action == 'setHeader') {
+                    obj.setHeader(historyRecord.column, historyRecord.newValue);
+                } else if (historyRecord.action == 'setComments') {
+                    obj.setComments(historyRecord.column, historyRecord.newValue[0], historyRecord.newValue[1]);
+                } else if (historyRecord.action == 'orderBy') {
+                    obj.updateOrderArrow(historyRecord.column, historyRecord.order);
+                    obj.updateOrder(historyRecord.rows);
+                } else if (historyRecord.action == 'setValue') {
+                    obj.setValue(historyRecord.records);
+                    // Redo for changes in cells
+                    for (var i = 0; i < historyRecord.records.length; i++) {
+                        if (historyRecord.oldStyle) {
+                            obj.resetStyle(historyRecord.newStyle);
+                        }
+                    }
+                    // Update selection
+                    if (historyRecord.selection) {
+                        obj.updateSelectionFromCoords(historyRecord.selection[0], historyRecord.selection[1], historyRecord.selection[2], historyRecord.selection[3]);
+                    }
+                }
+            }
+            obj.ignoreEvents = ignoreEvents;
+            obj.ignoreHistory = ignoreHistory;
+    
+            // Events
+            obj.dispatch('onredo', el, historyRecord);
+        }
+    
+        /**
+         * Get dropdown value from key
+         */
+        obj.getDropDownValue = function(column, key) {
+            var value = [];
+    
+            if (obj.options.columns[column] && obj.options.columns[column].source) {
+                // Create array from source
+                var combo = [];
+                var source = obj.options.columns[column].source;
+    
+                for (var i = 0; i < source.length; i++) {
+                    if (typeof(source[i]) == 'object') {
+                        combo[source[i].id] = source[i].name;
+                    } else {
+                        combo[source[i]] = source[i];
+                    }
+                }
+    
+                // Garante single multiple compatibily
+                var keys = ('' + key).split(';')
+    
+                for (var i = 0; i < keys.length; i++) {
+                    if (combo[keys[i]]) {
+                        value.push(combo[keys[i]]);
+                    }
+                }
+            } else {
+                console.error('Invalid column');
+            }
+    
+            return (value.length > 0) ? value.join('; ') : '';
+        }
+    
+        /**
+         * From starckoverflow contributions
+         */
+        obj.parseCSV = function(str, delimiter) {
+            // Remove last line break
+            str = str.replace(/\r?\n$|\r$|\n$/g, "");
+            // Last caracter is the delimiter
+            if (str.charCodeAt(str.length-1) == 9) {
+                str += "\0";
+            }
+            // user-supplied delimeter or default comma
+            delimiter = (delimiter || ",");
+    
+            var arr = [];
+            var quote = false;  // true means we're inside a quoted field
+            // iterate over each character, keep track of current row and column (of the returned array)
+            for (var row = 0, col = 0, c = 0; c < str.length; c++) {
+                var cc = str[c], nc = str[c+1];
+                arr[row] = arr[row] || [];
+                arr[row][col] = arr[row][col] || '';
+    
+                // If the current character is a quotation mark, and we're inside a quoted field, and the next character is also a quotation mark, add a quotation mark to the current column and skip the next character
+                if (cc == '"' && quote && nc == '"') { arr[row][col] += cc; ++c; continue; }  
+    
+                // If it's just one quotation mark, begin/end quoted field
+                if (cc == '"') { quote = !quote; continue; }
+    
+                // If it's a comma and we're not in a quoted field, move on to the next column
+                if (cc == delimiter && !quote) { ++col; continue; }
+    
+                // If it's a newline (CRLF) and we're not in a quoted field, skip the next character and move on to the next row and move to column 0 of that new row
+                if (cc == '\r' && nc == '\n' && !quote) { ++row; col = 0; ++c; continue; }
+    
+                // If it's a newline (LF or CR) and we're not in a quoted field, move on to the next row and move to column 0 of that new row
+                if (cc == '\n' && !quote) { ++row; col = 0; continue; }
+                if (cc == '\r' && !quote) { ++row; col = 0; continue; }
+    
+                // Otherwise, append the current character to the current column
+                arr[row][col] += cc;
+            }
+            return arr;
+        }
+    
+        obj.hash = function(str) {
+            var hash = 0, i, chr;
+    
+            if (str.length === 0) {
+                return hash;
+            } else {
+                for (i = 0; i < str.length; i++) {
+                  chr = str.charCodeAt(i);
+                  hash = ((hash << 5) - hash) + chr;
+                  hash |= 0;
+                }
+            }
+            return hash;
+        }
+    
+        obj.onafterchanges = function(el, records) {
+            // Events
+            obj.dispatch('onafterchanges', el, records);
+        }
+    
+        obj.destroy = function() {
+            jexcel.destroy(el);
+        }
+
+        /**
+         * Initialization method
+         */
+        obj.init = function() {
+            jexcel.current = obj;
+    
+            // Build handlers
+            if (typeof(jexcel.build) == 'function') {
+                if (obj.options.root) {
+                    jexcel.build(obj.options.root);
+                } else {
+                    jexcel.build(document);
+                    jexcel.build = null;
+                }
+            }
+    
+            // Event
+            el.setAttribute('tabindex', 1);
+            el.addEventListener('focus', function(e) {
+                if (jexcel.current && ! obj.selectedCell) {
+                    obj.updateSelectionFromCoords(0,0,0,0);
+                    obj.left();
+                }
+            });
+
+            // Load the table data based on an CSV file
+            if (obj.options.csv) {
+                // Loading
+                if (obj.options.loadingSpin == true) {
+                    jSuites.loading.show();
+                }
+    
+                // Load CSV file
+                jSuites.ajax({
+                    url: obj.options.csv,
+                    method: 'GET',
+                    dataType: 'text',
+                    success: function(result) {
+                        // Convert data
+                        var newData = obj.parseCSV(result, obj.options.csvDelimiter)
+    
+                        // Headers
+                        if (obj.options.csvHeaders == true && newData.length > 0) {
+                            var headers = newData.shift();
+                            for(var i = 0; i < headers.length; i++) {
+                                if (! obj.options.columns[i]) {
+                                    obj.options.columns[i] = { type:'text', align:obj.options.defaultColAlign, width:obj.options.defaultColWidth };
+                                }
+                                // Precedence over pre-configurated titles
+                                if (typeof obj.options.columns[i].title === 'undefined') {
+                                  obj.options.columns[i].title = headers[i];
+                                }
+                            }
+                        }
+                        // Data
+                        obj.options.data = newData;
+                        // Prepare table
+                        obj.prepareTable();
+                        // Hide spin
+                        if (obj.options.loadingSpin == true) {
+                            jSuites.loading.hide();
+                        }
+                    }
+                });
+            } else if (obj.options.url) {
+                // Loading
+                if (obj.options.loadingSpin == true) {
+                    jSuites.loading.show();
+                }
+    
+                jSuites.ajax({
+                    url: obj.options.url,
+                    method: 'GET',
+                    dataType: 'json',
+                    success: function(result) {
+                        // Data
+                        obj.options.data = (result.data) ? result.data : result;
+                        // Prepare table
+                        obj.prepareTable();
+                        // Hide spin
+                        if (obj.options.loadingSpin == true) {
+                            jSuites.loading.hide();
+                        }
+                    }
+                });
+            } else {
+                // Prepare table
+                obj.prepareTable();
+            }
+        }
+    
+        // Context menu
+        if (options && options.contextMenu != null) {
+            obj.options.contextMenu = options.contextMenu;
+        } else {
+            obj.options.contextMenu = function(el, x, y, e) {
+                var items = [];
+    
+                if (y == null) {
+                    // Insert a new column
+                    if (obj.options.allowInsertColumn == true) {
+                        items.push({
+                            title:obj.options.text.insertANewColumnBefore,
+                            onclick:function() {
+                                obj.insertColumn(1, parseInt(x), 1);
+                            }
+                        });
+                    }
+    
+                    if (obj.options.allowInsertColumn == true) {
+                        items.push({
+                            title:obj.options.text.insertANewColumnAfter,
+                            onclick:function() {
+                                obj.insertColumn(1, parseInt(x), 0);
+                            }
+                        });
+                    }
+    
+                    // Delete a column
+                    if (obj.options.allowDeleteColumn == true) {
+                        items.push({
+                            title:obj.options.text.deleteSelectedColumns,
+                            onclick:function() {
+                                obj.deleteColumn(obj.getSelectedColumns().length ? undefined : parseInt(x));
+                            }
+                        });
+                    }
+    
+                    // Rename column
+                    if (obj.options.allowRenameColumn == true) {
+                        items.push({
+                            title:obj.options.text.renameThisColumn,
+                            onclick:function() {
+                                obj.setHeader(x);
+                            }
+                        });
+                    }
+    
+                    // Sorting
+                    if (obj.options.columnSorting == true) {
+                        // Line
+                        items.push({ type:'line' });
+    
+                        items.push({
+                            title:obj.options.text.orderAscending,
+                            onclick:function() {
+                                obj.orderBy(x, 0);
+                            }
+                        });
+                        items.push({
+                            title:obj.options.text.orderDescending,
+                            onclick:function() {
+                                obj.orderBy(x, 1);
+                            }
+                        });
+                    }
+                } else {
+                    // Insert new row
+                    if (obj.options.allowInsertRow == true) {
+                        items.push({
+                            title:obj.options.text.insertANewRowBefore,
+                            onclick:function() {
+                                obj.insertRow(1, parseInt(y), 1);
+                            }
+                        });
+    
+                        items.push({
+                            title:obj.options.text.insertANewRowAfter,
+                            onclick:function() {
+                                obj.insertRow(1, parseInt(y));
+                            }
+                        });
+                    }
+    
+                    if (obj.options.allowDeleteRow == true) {
+                        items.push({
+                            title:obj.options.text.deleteSelectedRows,
+                            onclick:function() {
+                                obj.deleteRow(obj.getSelectedRows().length ? undefined : parseInt(y));
+                            }
+                        });
+                    }
+    
+                    if (x) {
+                        if (obj.options.allowComments == true) {
+                            items.push({ type:'line' });
+    
+                            var title = obj.records[y][x].getAttribute('title') || '';
+
+                            items.push({
+                                title: title ? obj.options.text.editComments : obj.options.text.addComments,
+                                onclick:function() {
+                                    var comment = prompt(obj.options.text.comments, title);
+                                    if (comment) {
+                                        obj.setComments([ x, y ], comment);
+                                    }
+                                }
+                            });
+    
+                            if (title) {
+                                items.push({
+                                    title:obj.options.text.clearComments,
+                                    onclick:function() {
+                                        obj.setComments([ x, y ], '');
+                                    }
+                                });
+                            }
+                        }
+                    }
+                }
+    
+                // Line
+                items.push({ type:'line' });
+    
+                // Copy
+                items.push({
+                    title:obj.options.text.copy,
+                    shortcut:'Ctrl + C',
+                    onclick:function() {
+                        obj.copy(true);
+                    }
+                });
+    
+                // Paste
+                if (navigator && navigator.clipboard) {
+                    items.push({
+                        title:obj.options.text.paste,
+                        shortcut:'Ctrl + V',
+                        onclick:function() {
+                            if (obj.selectedCell) {
+                                navigator.clipboard.readText().then(function(text) {
+                                    if (text) {
+                                        jexcel.current.paste(obj.selectedCell[0], obj.selectedCell[1], text);
+                                    }
+                                });
+                            }
+                        }
+                    });
+                }
+    
+                // Save
+                if (obj.options.allowExport) {
+                    items.push({
+                        title: obj.options.text.saveAs,
+                        shortcut: 'Ctrl + S',
+                        onclick: function () {
+                            obj.download();
+                        }
+                    });
+                }
+    
+                // About
+                if (obj.options.about) {
+                    items.push({
+                        title:obj.options.text.about,
+                        onclick:function() {
+                            alert(obj.options.about);
+                        }
+                    });
+                }
+    
+                return items;
+            }
+        }
+    
+        obj.scrollControls = function(e) {
+            obj.wheelControls();
+
+            if (obj.options.freezeColumns > 0 && obj.content.scrollLeft != scrollLeft) {
+                obj.updateFreezePosition();
+            }
+
+            // Close editor
+            if (obj.options.lazyLoading == true || obj.options.tableOverflow == true) {
+                if (obj.edition && e.target.className.substr(0,9) != 'jdropdown') {
+                    obj.closeEditor(obj.edition[0], true);
+                }
+            }
+        }
+
+        obj.wheelControls = function(e) {
+            if (obj.options.lazyLoading == true) {
+                if (jexcel.timeControlLoading == null) {
+                    jexcel.timeControlLoading = setTimeout(function() {
+                        if (obj.content.scrollTop + obj.content.clientHeight >= obj.content.scrollHeight) {
+                            if (obj.loadDown()) {
+                                if (obj.content.scrollTop + obj.content.clientHeight > obj.content.scrollHeight - 10) {
+                                    obj.content.scrollTop = obj.content.scrollTop - obj.content.clientHeight;
+                                }
+                                obj.updateCornerPosition();
+                            }
+                        } else if (obj.content.scrollTop <= obj.content.clientHeight) {
+                            if (obj.loadUp()) {
+                                if (obj.content.scrollTop < 10) {
+                                    obj.content.scrollTop = obj.content.scrollTop + obj.content.clientHeight;
+                                }
+                                obj.updateCornerPosition();
+                            }
+                        }
+    
+                        jexcel.timeControlLoading = null;
+                    }, 100);
+                }
+            }
+        }
+
+        // Get width of all freezed cells together
+        obj.getFreezeWidth = function() {
+            var width = 0;
+            if (obj.options.freezeColumns > 0) {
+                for (var i = 0; i < obj.options.freezeColumns; i++) {
+                    width += parseInt(obj.options.columns[i].width);
+                }
+            }
+            return width;
+        }
+
+        var scrollLeft = 0;
+
+        obj.updateFreezePosition = function() {
+            scrollLeft = obj.content.scrollLeft;
+            var width = 0;
+            if (scrollLeft > 50) {
+                for (var i = 0; i < obj.options.freezeColumns; i++) {
+                    if (i > 0) {
+                        width += parseInt(obj.options.columns[i-1].width);
+                    }
+                    obj.headers[i].classList.add('jexcel_freezed');
+                    obj.headers[i].style.left = width + 'px';
+                    for (var j = 0; j < obj.rows.length; j++) {
+                        if (obj.rows[j] && obj.records[j][i]) {
+                            var shifted = (scrollLeft + (i > 0 ? obj.records[j][i-1].style.width : 0)) - 51 + 'px';
+                            obj.records[j][i].classList.add('jexcel_freezed');
+                            obj.records[j][i].style.left = shifted;
+                        }
+                    }
+                }
+            } else {
+                for (var i = 0; i < obj.options.freezeColumns; i++) {
+                    obj.headers[i].classList.remove('jexcel_freezed');
+                    obj.headers[i].style.left = '';
+                    for (var j = 0; j < obj.rows.length; j++) {
+                        if (obj.records[j][i]) {
+                            obj.records[j][i].classList.remove('jexcel_freezed');
+                            obj.records[j][i].style.left = '';
+                        }
+                    }
+                }
+            }
+
+            // Place the corner in the correct place
+            obj.updateCornerPosition();
+        }
+
+        el.addEventListener("DOMMouseScroll", obj.wheelControls);
+        el.addEventListener("mousewheel", obj.wheelControls);
+    
+        el.jexcel = obj;
+    
+        obj.init();
+    
+        return obj;
+    });
+    
+    jexcel.current = null;
+    jexcel.timeControl = null;
+    jexcel.timeControlLoading = null;
+    
+    jexcel.destroy = function(element, destroyEventHandlers) {
+        if (element.jexcel) {
+            var root = element.jexcel.options.root ? element.jexcel.options.root : document;
+            element.removeEventListener("DOMMouseScroll", element.jexcel.scrollControls);
+            element.removeEventListener("mousewheel", element.jexcel.scrollControls);
+            element.jexcel = null;
+            element.innerHTML = '';
+    
+            if (destroyEventHandlers) {
+                root.removeEventListener("mouseup", jexcel.mouseUpControls);
+                root.removeEventListener("mousedown", jexcel.mouseDownControls);
+                root.removeEventListener("mousemove", jexcel.mouseMoveControls);
+                root.removeEventListener("mouseover", jexcel.mouseOverControls);
+                root.removeEventListener("dblclick", jexcel.doubleClickControls);
+                root.removeEventListener("paste", jexcel.pasteControls);
+                root.removeEventListener("contextmenu", jexcel.contextMenuControls);
+                root.removeEventListener("touchstart", jexcel.touchStartControls);
+                root.removeEventListener("touchend", jexcel.touchEndControls);
+                root.removeEventListener("touchcancel", jexcel.touchEndControls);
+                document.removeEventListener("keydown", jexcel.keyDownControls);
+                jexcel = null;
+            }
+        }
+    }
+    
+    jexcel.build = function(root) {
+        root.addEventListener("mouseup", jexcel.mouseUpControls);
+        root.addEventListener("mousedown", jexcel.mouseDownControls);
+        root.addEventListener("mousemove", jexcel.mouseMoveControls);
+        root.addEventListener("mouseover", jexcel.mouseOverControls);
+        root.addEventListener("dblclick", jexcel.doubleClickControls);
+        root.addEventListener("paste", jexcel.pasteControls);
+        root.addEventListener("contextmenu", jexcel.contextMenuControls);
+        root.addEventListener("touchstart", jexcel.touchStartControls);
+        root.addEventListener("touchend", jexcel.touchEndControls);
+        root.addEventListener("touchcancel", jexcel.touchEndControls);
+        root.addEventListener("touchmove", jexcel.touchEndControls);
+        document.addEventListener("keydown", jexcel.keyDownControls);
+    }
+    
+    /**
+     * Events
+     */
+    jexcel.keyDownControls = function(e) {
+        if (jexcel.current) {
+            if (jexcel.current.edition) {
+                if (e.which == 27) {
+                    // Escape
+                    if (jexcel.current.edition) {
+                        // Exit without saving
+                        jexcel.current.closeEditor(jexcel.current.edition[0], false);
+                    }
+                    e.preventDefault();
+                } else if (e.which == 13) {
+                    // Enter
+                    if (jexcel.current.options.columns[jexcel.current.edition[2]].type == 'calendar') {
+                        jexcel.current.closeEditor(jexcel.current.edition[0], true);
+                    } else if (jexcel.current.options.columns[jexcel.current.edition[2]].type == 'dropdown' ||
+                               jexcel.current.options.columns[jexcel.current.edition[2]].type == 'autocomplete') {
+                        // Do nothing
+                    } else {
+                        // Alt enter -> do not close editor
+                        if ((jexcel.current.options.wordWrap == true ||
+                             jexcel.current.options.columns[jexcel.current.edition[2]].wordWrap == true ||
+                             jexcel.current.options.data[jexcel.current.edition[3]][jexcel.current.edition[2]].length > 200) && e.altKey) {
+                            // Add new line to the editor
+                            var editorTextarea = jexcel.current.edition[0].children[0];
+                            var editorValue = jexcel.current.edition[0].children[0].value;
+                            var editorIndexOf = editorTextarea.selectionStart;
+                            editorValue = editorValue.slice(0, editorIndexOf) + "\n" + editorValue.slice(editorIndexOf);
+                            editorTextarea.value = editorValue;
+                            editorTextarea.focus();
+                            editorTextarea.selectionStart = editorIndexOf + 1;
+                            editorTextarea.selectionEnd = editorIndexOf + 1;
+                        } else {
+                            jexcel.current.edition[0].children[0].blur();
+                        }
+                    }
+                } else if (e.which == 9) {
+                    // Tab
+                    if (jexcel.current.options.columns[jexcel.current.edition[2]].type == 'calendar') {
+                        jexcel.current.closeEditor(jexcel.current.edition[0], true);
+                    } else {
+                        jexcel.current.edition[0].children[0].blur();
+                    }
+                }
+            }
+    
+            if (! jexcel.current.edition && jexcel.current.selectedCell) {
+                // Which key
+                if (e.which == 37) {
+                    jexcel.current.left(e.shiftKey, e.ctrlKey);
+                    e.preventDefault();
+                } else if (e.which == 39) {
+                    jexcel.current.right(e.shiftKey, e.ctrlKey);
+                    e.preventDefault();
+                } else if (e.which == 38) {
+                    jexcel.current.up(e.shiftKey, e.ctrlKey);
+                    e.preventDefault();
+                } else if (e.which == 40) {
+                    jexcel.current.down(e.shiftKey, e.ctrlKey);
+                    e.preventDefault();
+                } else if (e.which == 36) {
+                    jexcel.current.first(e.shiftKey, e.ctrlKey);
+                    e.preventDefault();
+                } else if (e.which == 35) {
+                    jexcel.current.last(e.shiftKey, e.ctrlKey);
+                    e.preventDefault();
+                } else if (e.which == 32) {
+                    if (jexcel.current.options.editable == true) {
+                        jexcel.current.setCheckRadioValue();
+                    }
+                    e.preventDefault();
+                } else if (e.which == 46) {
+                    // Delete
+                    if (jexcel.current.options.editable == true) {
+                        if (jexcel.current.selectedRow) {
+                            if (jexcel.current.options.allowDeleteRow == true) {
+                                if (confirm(jexcel.current.options.text.areYouSureToDeleteTheSelectedRows)) {
+                                    jexcel.current.deleteRow();
+                                }
+                            }
+                        } else if (jexcel.current.selectedHeader) {
+                            if (jexcel.current.options.allowDeleteColumn == true) {
+                                if (confirm(jexcel.current.options.text.areYouSureToDeleteTheSelectedColumns)) {
+                                    jexcel.current.deleteColumn();
+                                }
+                            }
+                        } else {
+                            // Change value
+                            jexcel.current.setValue(jexcel.current.highlighted, '');
+                        }
+                    }
+                } else if (e.which == 13) {
+                    // Move cursor
+                    if (e.shiftKey) {
+                        jexcel.current.up();
+                    } else {
+                        if (jexcel.current.options.allowInsertRow == true) {
+                            if (jexcel.current.options.allowManualInsertRow == true) {
+                                if (jexcel.current.selectedCell[1] == jexcel.current.options.data.length - 1) {
+                                    // New record in case selectedCell in the last row
+                                    jexcel.current.insertRow();
+                                }
+                            }
+                        }
+    
+                        jexcel.current.down();
+                    }
+                    e.preventDefault();
+                } else if (e.which == 9) {
+                    // Tab
+                    if (e.shiftKey) {
+                        jexcel.current.left();
+                    } else {
+                        if (jexcel.current.options.allowInsertColumn == true) {
+                            if (jexcel.current.options.allowManualInsertColumn == true) {
+                                if (jexcel.current.selectedCell[0] == jexcel.current.options.data[0].length - 1) {
+                                    // New record in case selectedCell in the last column
+                                    jexcel.current.insertColumn();
+                                }
+                            }
+                        }
+    
+                        jexcel.current.right();
+                    }
+                    e.preventDefault();
+                } else {
+                    if ((e.ctrlKey || e.metaKey) && ! e.shiftKey) {
+                        if (e.which == 65) {
+                            // Ctrl + A
+                            jexcel.current.selectAll();
+                            e.preventDefault();
+                        } else if (e.which == 83) {
+                            // Ctrl + S
+                            jexcel.current.download();
+                            e.preventDefault();
+                        } else if (e.which == 89) {
+                            // Ctrl + Y
+                            jexcel.current.redo();
+                            e.preventDefault();
+                        } else if (e.which == 90) {
+                            // Ctrl + Z
+                            jexcel.current.undo();
+                            e.preventDefault();
+                        } else if (e.which == 67) {
+                            // Ctrl + C
+                            jexcel.current.copy(true);
+                            e.preventDefault();
+                        } else if (e.which == 67) {
+                            // Ctrl + C
+                            jexcel.current.copy(true);
+                            e.preventDefault();
+                        } else if (e.which == 88) {
+                            // Ctrl + X
+                            if (jexcel.current.options.editable == true) {
+                                jexcel.cutControls();
+                            } else {
+                                jexcel.copyControls();
+                            }
+                            e.preventDefault();
+                        } else if (e.which == 86) {
+                            // Ctrl + V
+                            jexcel.pasteControls();
+                        }
+                    } else {
+                        if (jexcel.current.selectedCell) {
+                            if (jexcel.current.options.editable == true) {
+                                var rowId = jexcel.current.selectedCell[1];
+                                var columnId = jexcel.current.selectedCell[0];
+    
+                                // If is not readonly
+                                if (jexcel.current.options.columns[columnId].type != 'readonly') {
+                                    // Characters able to start a edition
+                                    if (e.keyCode == 32) {
+                                        // Space
+                                        if (jexcel.current.options.columns[columnId].type == 'checkbox' ||
+                                            jexcel.current.options.columns[columnId].type == 'radio') {
+                                            e.preventDefault();
+                                        } else {
+                                            // Start edition
+                                            jexcel.current.openEditor(jexcel.current.records[rowId][columnId], true);
+                                        }
+                                    } else if (e.keyCode == 113) {
+                                        // Start edition with current content F2
+                                        jexcel.current.openEditor(jexcel.current.records[rowId][columnId], false);
+                                    } else if ((e.keyCode == 8) ||
+                                               (e.keyCode >= 48 && e.keyCode <= 57) ||
+                                               (e.keyCode >= 96 && e.keyCode <= 111) ||
+                                               (e.keyCode == 187) ||
+                                               (e.keyCode == 189) ||
+                                               ((String.fromCharCode(e.keyCode) == e.key || String.fromCharCode(e.keyCode).toLowerCase() == e.key.toLowerCase()) && jexcel.validLetter(String.fromCharCode(e.keyCode)))) {
+                                        // Start edition
+                                        jexcel.current.openEditor(jexcel.current.records[rowId][columnId], true);
+                                        // Prevent entries in the calendar
+                                        if (jexcel.current.options.columns[columnId].type == 'calendar') {
+                                            e.preventDefault();
+                                        }
+                                    }
+                                }
+                            }
+                        }
+                    }
+                }
+            } else {
+                if (e.target.classList.contains('jexcel_search')) {
+                    if (jexcel.timeControl) {
+                        clearTimeout(jexcel.timeControl);
+                    }
+    
+                    jexcel.timeControl = setTimeout(function() {
+                        jexcel.current.search(e.target.value);
+                    }, 200);
+                }
+            }
+        }
+    }
+    
+    jexcel.isMouseAction = false;
+    
+    jexcel.mouseDownControls = function(e) {
+        e = e || window.event;
+        if (e.buttons) {
+            var mouseButton = e.buttons;
+        } else if (e.button) {
+            var mouseButton = e.button;
+        } else {
+            var mouseButton = e.which;
+        }
+
+        // Get elements
+        var jexcelTable = jexcel.getElement(e.target);
+
+        if (jexcelTable[0]) {
+            if (jexcel.current != jexcelTable[0].jexcel) {
+                if (jexcel.current) {
+                    if (jexcel.current.edition) {
+                        jexcel.current.closeEditor(jexcel.current.edition[0], true);
+                    }
+                    jexcel.current.resetSelection();
+                }
+                jexcel.current = jexcelTable[0].jexcel;
+            }
+        } else {
+            if (jexcel.current) {
+                if (jexcel.current.edition) {
+                    jexcel.current.closeEditor(jexcel.current.edition[0], true);
+                }
+                
+                jexcel.current.resetSelection(true);
+                jexcel.current = null;
+            }
+        }
+    
+        if (jexcel.current && mouseButton == 1) {
+            if (e.target.classList.contains('jexcel_selectall')) {
+                if (jexcel.current) {
+                    jexcel.current.selectAll();
+                }
+            } else if (e.target.classList.contains('jexcel_corner')) {
+                if (jexcel.current.options.editable == true) {
+                    jexcel.current.selectedCorner = true;
+                }
+            } else {
+                // Header found
+                if (jexcelTable[1] == 1) {
+                    var columnId = e.target.getAttribute('data-x');
+                    if (columnId) {
+                        // Update cursor
+                        var info = e.target.getBoundingClientRect();
+                        if (jexcel.current.options.columnResize == true && info.width - e.offsetX < 6) {
+                            // Resize helper
+                            jexcel.current.resizing = {
+                                mousePosition: e.pageX,
+                                column: columnId,
+                                width: info.width,
+                            };
+    
+                            // Border indication
+                            jexcel.current.headers[columnId].classList.add('resizing');
+                            for (var j = 0; j < jexcel.current.records.length; j++) {
+                                if (jexcel.current.records[j][columnId]) {
+                                    jexcel.current.records[j][columnId].classList.add('resizing');
+                                }
+                            }
+                        } else if (jexcel.current.options.columnDrag == true && info.height - e.offsetY < 6) {
+                            if (jexcel.current.isColMerged(columnId).length) {
+                                console.error('JEXCEL: This column is part of a merged cell.');
+                            } else {
+                                // Reset selection
+                                jexcel.current.resetSelection();
+                                // Drag helper
+                                jexcel.current.dragging = {
+                                    element: e.target,
+                                    column:columnId,
+                                    destination:columnId,
+                                };
+                                // Border indication
+                                jexcel.current.headers[columnId].classList.add('dragging');
+                                for (var j = 0; j < jexcel.current.records.length; j++) {
+                                    if (jexcel.current.records[j][columnId]) {
+                                        jexcel.current.records[j][columnId].classList.add('dragging');
+                                    }
+                                }
+                            }
+                        } else {
+                            if (jexcel.current.selectedHeader && (e.shiftKey || e.ctrlKey)) {
+                                var o = jexcel.current.selectedHeader;
+                                var d = columnId;
+                            } else {
+                                // Press to rename
+                                if (jexcel.current.selectedHeader == columnId && jexcel.current.options.allowRenameColumn == true) {
+                                    jexcel.timeControl = setTimeout(function() {
+                                        jexcel.current.setHeader(columnId);
+                                    }, 800);
+                                }
+    
+                                // Keep track of which header was selected first
+                                jexcel.current.selectedHeader = columnId;
+    
+                                // Update selection single column
+                                var o = columnId;
+                                var d = columnId;
+                            }
+    
+                            // Update selection
+                            jexcel.current.updateSelectionFromCoords(o, 0, d, jexcel.current.options.data.length - 1);
+                        }
+                    } else {
+                        if (e.target.parentNode.classList.contains('jexcel_nested')) {
+                            if (e.target.getAttribute('data-column')) {
+                                var column = e.target.getAttribute('data-column').split(',');
+                                var c1 = parseInt(column[0]);
+                                var c2 = parseInt(column[column.length-1]);
+                            } else {
+                                var c1 = 0;
+                                var c2 = jexcel.current.options.columns.length - 1;
+                            }
+                            jexcel.current.updateSelectionFromCoords(c1, 0, c2, jexcel.current.options.data.length - 1);
+                        }
+                    }
+                } else {
+                    jexcel.current.selectedHeader = false;
+                }
+    
+                // Body found
+                if (jexcelTable[1] == 2) {
+                    var rowId = e.target.getAttribute('data-y');
+                    
+                    if (e.target.classList.contains('jexcel_row')) {
+                        var info = e.target.getBoundingClientRect();
+                        if (jexcel.current.options.rowResize == true && info.height - e.offsetY < 6) {
+                            // Resize helper
+                            jexcel.current.resizing = {
+                                element: e.target.parentNode,
+                                mousePosition: e.pageY,
+                                row: rowId,
+                                height: info.height,
+                            };
+                            // Border indication
+                            e.target.parentNode.classList.add('resizing');
+                        } else if (jexcel.current.options.rowDrag == true && info.width - e.offsetX < 6) {
+                            if (jexcel.current.isRowMerged(rowId).length) {
+                                console.error('JEXCEL: This row is part of a merged cell');
+                            } else if (jexcel.current.options.search == true && jexcel.current.results) {
+                                console.error('JEXCEL: Please clear your search before perform this action');
+                            } else {
+                                // Reset selection
+                                jexcel.current.resetSelection();
+                                // Drag helper
+                                jexcel.current.dragging = {
+                                    element: e.target.parentNode,
+                                    row:rowId,
+                                    destination:rowId,
+                                };
+                                // Border indication
+                                e.target.parentNode.classList.add('dragging');
+                            }
+                        } else {
+                            if (jexcel.current.selectedRow && (e.shiftKey || e.ctrlKey)) {
+                                var o = jexcel.current.selectedRow;
+                                var d = rowId;
+                            } else {
+                                // Keep track of which header was selected first
+                                jexcel.current.selectedRow = rowId;
+    
+                                // Update selection single column
+                                var o = rowId;
+                                var d = rowId;
+                            }
+    
+                            // Update selection
+                            jexcel.current.updateSelectionFromCoords(0, o, jexcel.current.options.data[0].length - 1, d);
+                        }
+                    } else {
+                        // Jclose
+                        if (e.target.classList.contains('jclose') && e.target.clientWidth - e.offsetX < 50 && e.offsetY < 50) {
+                            jexcel.current.closeEditor(jexcel.current.edition[0], true);
+                        } else {
+                            var getCellCoords = function(element) {
+                                var x = element.getAttribute('data-x');
+                                var y = element.getAttribute('data-y');
+                                if (x && y) {
+                                    return [x, y];
+                                } else {
+                                    if (element.parentNode) {
+                                        return getCellCoords(element.parentNode);
+                                    }
+                                }
+                            };
+
+                            var position = getCellCoords(e.target);
+                            if (position) {
+
+                                var columnId = position[0];
+                                var rowId = position[1];
+                                // Close edition
+                                if (jexcel.current.edition) {
+                                    if (jexcel.current.edition[2] != columnId || jexcel.current.edition[3] != rowId) {
+                                        jexcel.current.closeEditor(jexcel.current.edition[0], true);
+                                    }
+                                }
+
+                                if (! jexcel.current.edition) {
+                                    // Update cell selection
+                                    if (e.shiftKey) {
+                                        jexcel.current.updateSelectionFromCoords(jexcel.current.selectedCell[0], jexcel.current.selectedCell[1], columnId, rowId);
+                                    } else {
+                                        jexcel.current.updateSelectionFromCoords(columnId, rowId);
+                                    }
+                                }
+
+                                // No full row selected
+                                jexcel.current.selectedHeader = null;
+                                jexcel.current.selectedRow = null;
+                            }
+                        }
+                    }
+                } else {
+                    jexcel.current.selectedRow = false;
+                }
+    
+                // Pagination
+                if (e.target.classList.contains('jexcel_page')) {
+                    if (e.target.innerText == '<') {
+                        jexcel.current.page(0);
+                    } else if (e.target.innerText == '>') {
+                        jexcel.current.page(e.target.getAttribute('title') - 1);
+                    } else {
+                        jexcel.current.page(e.target.innerText - 1);
+                    }
+                }
+            }
+    
+            if (jexcel.current.edition) {
+                jexcel.isMouseAction = false;
+            } else {
+                jexcel.isMouseAction = true;
+            }
+        } else {
+            jexcel.isMouseAction = false;
+        }
+    }
+    
+    jexcel.mouseUpControls = function(e) {
+        if (jexcel.current) {
+            // Update cell size
+            if (jexcel.current.resizing) {
+                // Columns to be updated
+                if (jexcel.current.resizing.column) {
+                    // New width
+                    var newWidth = jexcel.current.colgroup[jexcel.current.resizing.column].getAttribute('width');
+                    // Columns
+                    var columns = jexcel.current.getSelectedColumns();
+                    if (columns.length > 1) {
+                        var currentWidth = [];
+                        for (var i = 0; i < columns.length; i++) {
+                            currentWidth.push(parseInt(jexcel.current.colgroup[columns[i]].getAttribute('width')));
+                        }
+                        // Previous width
+                        var index = columns.indexOf(parseInt(jexcel.current.resizing.column));
+                        currentWidth[index] = jexcel.current.resizing.width;
+                        jexcel.current.setWidth(columns, newWidth, currentWidth);
+                    } else {
+                        jexcel.current.setWidth(jexcel.current.resizing.column, newWidth, jexcel.current.resizing.width);
+                    }
+                    // Remove border
+                    jexcel.current.headers[jexcel.current.resizing.column].classList.remove('resizing');
+                    for (var j = 0; j < jexcel.current.records.length; j++) {
+                        if (jexcel.current.records[j][jexcel.current.resizing.column]) {
+                            jexcel.current.records[j][jexcel.current.resizing.column].classList.remove('resizing');
+                        }
+                    }
+                } else {
+                    // Remove Class
+                    jexcel.current.rows[jexcel.current.resizing.row].children[0].classList.remove('resizing');
+                    var newHeight = jexcel.current.rows[jexcel.current.resizing.row].getAttribute('height');
+                    jexcel.current.setHeight(jexcel.current.resizing.row, newHeight, jexcel.current.resizing.height);
+                    // Remove border
+                    jexcel.current.resizing.element.classList.remove('resizing');
+                }
+                // Reset resizing helper
+                jexcel.current.resizing = null;
+            } else if (jexcel.current.dragging) {
+                // Reset dragging helper
+                if (jexcel.current.dragging) {
+                    if (jexcel.current.dragging.column) {
+                        // Target
+                        var columnId = e.target.getAttribute('data-x');
+                        // Remove move style
+                        jexcel.current.headers[jexcel.current.dragging.column].classList.remove('dragging');
+                        for (var j = 0; j < jexcel.current.rows.length; j++) {
+                            if (jexcel.current.records[j][jexcel.current.dragging.column]) {
+                                jexcel.current.records[j][jexcel.current.dragging.column].classList.remove('dragging');
+                            }
+                        }
+                        for (var i = 0; i < jexcel.current.headers.length; i++) {
+                            jexcel.current.headers[i].classList.remove('dragging-left');
+                            jexcel.current.headers[i].classList.remove('dragging-right');
+                        }
+                        // Update position
+                        if (columnId) {
+                            if (jexcel.current.dragging.column != jexcel.current.dragging.destination) {
+                                jexcel.current.moveColumn(jexcel.current.dragging.column, jexcel.current.dragging.destination);
+                            }
+                        }
+                    } else {
+                        if (jexcel.current.dragging.element.nextSibling) {
+                            var position = parseInt(jexcel.current.dragging.element.nextSibling.getAttribute('data-y'));
+                            if (jexcel.current.dragging.row < position) {
+                                position -= 1;
+                            }
+                        } else {
+                            var position = parseInt(jexcel.current.dragging.element.previousSibling.getAttribute('data-y'));
+                        }
+                        if (jexcel.current.dragging.row != position) {
+                            jexcel.current.moveRow(jexcel.current.dragging.row, position, true);
+                        }
+                        jexcel.current.dragging.element.classList.remove('dragging');
+                    }
+                    jexcel.current.dragging = null;
+                }
+            } else {
+                // Close any corner selection
+                if (jexcel.current.selectedCorner) {
+                    jexcel.current.selectedCorner = false;
+    
+                    // Data to be copied
+                    if (jexcel.current.selection.length > 0) {
+                        // Copy data
+                        jexcel.current.copyData(jexcel.current.selection[0], jexcel.current.selection[jexcel.current.selection.length - 1]);
+    
+                        // Remove selection
+                        jexcel.current.removeCopySelection();
+                    }
+                }
+            }
+        }
+
+        // Clear any time control
+        if (jexcel.timeControl) {
+            clearTimeout(jexcel.timeControl);
+            jexcel.timeControl = null;
+        }
+    
+        // Mouse up
+        jexcel.isMouseAction = false;
+    }
+    
+    // Mouse move controls
+    jexcel.mouseMoveControls = function(e) {
+        e = e || window.event;
+        if (e.buttons) {
+            var mouseButton = e.buttons;
+        } else if (e.button) {
+            var mouseButton = e.button;
+        } else {
+            var mouseButton = e.which;
+        }
+    
+        if (! mouseButton) {
+            jexcel.isMouseAction = false;
+        }
+    
+        if (jexcel.current) {
+            if (jexcel.isMouseAction == true) {
+                // Resizing is ongoing
+                if (jexcel.current.resizing) {
+                    if (jexcel.current.resizing.column) {
+                        var width = e.pageX - jexcel.current.resizing.mousePosition;
+    
+                        if (jexcel.current.resizing.width + width > 0) {
+                            var tempWidth = jexcel.current.resizing.width + width;
+                            jexcel.current.colgroup[jexcel.current.resizing.column].setAttribute('width', tempWidth);
+    
+                            jexcel.current.updateCornerPosition();
+                        }
+                    } else {
+                        var height = e.pageY - jexcel.current.resizing.mousePosition;
+    
+                        if (jexcel.current.resizing.height + height > 0) {
+                            var tempHeight = jexcel.current.resizing.height + height;
+                            jexcel.current.rows[jexcel.current.resizing.row].setAttribute('height', tempHeight);
+    
+                            jexcel.current.updateCornerPosition();
+                        }
+                    }
+                }
+            } else {
+                var x = e.target.getAttribute('data-x');
+                var y = e.target.getAttribute('data-y');
+                var rect = e.target.getBoundingClientRect();
+    
+                if (jexcel.current.cursor) {
+                    jexcel.current.cursor.style.cursor = '';
+                    jexcel.current.cursor = null;
+                }
+
+                if (e.target.parentNode.parentNode && e.target.parentNode.parentNode.className) {
+                    if (e.target.parentNode.parentNode.classList.contains('resizable')) {
+                        if (e.target && x && ! y && (rect.width - (e.clientX - rect.left) < 6)) {
+                            jexcel.current.cursor = e.target;
+                            jexcel.current.cursor.style.cursor = 'col-resize';
+                        } else if (e.target && ! x && y && (rect.height - (e.clientY - rect.top) < 6)) {
+                            jexcel.current.cursor = e.target;
+                            jexcel.current.cursor.style.cursor = 'row-resize';
+                        }
+                    }
+    
+                    if (e.target.parentNode.parentNode.classList.contains('draggable')) {
+                        if (e.target && ! x && y && (rect.width - (e.clientX - rect.left) < 6)) {
+                            jexcel.current.cursor = e.target;
+                            jexcel.current.cursor.style.cursor = 'move';
+                        } else if (e.target && x && ! y && (rect.height - (e.clientY - rect.top) < 6)) {
+                            jexcel.current.cursor = e.target;
+                            jexcel.current.cursor.style.cursor = 'move';
+                        }
+                    }
+                }
+            }
+        }
+    }
+    
+    jexcel.mouseOverControls = function(e) {
+        e = e || window.event;
+        if (e.buttons) {
+            var mouseButton = e.buttons;
+        } else if (e.button) {
+            var mouseButton = e.button;
+        } else {
+            var mouseButton = e.which;
+        }
+    
+        if (! mouseButton) {
+            jexcel.isMouseAction = false;
+        }
+    
+        if (jexcel.current && jexcel.isMouseAction == true) {
+            // Get elements
+            var jexcelTable = jexcel.getElement(e.target);
+    
+            if (jexcelTable[0]) {
+                // Avoid cross reference
+                if (jexcel.current != jexcelTable[0].jexcel) {
+                    if (jexcel.current) {
+                        return false;
+                    }
+                }
+    
+                var columnId = e.target.getAttribute('data-x');
+                var rowId = e.target.getAttribute('data-y');
+    
+                if (jexcel.current.dragging) {
+                    if (jexcel.current.dragging.column) {
+                        if (columnId) {
+                            if (jexcel.current.isColMerged(columnId).length) {
+                                console.error('JEXCEL: This column is part of a merged cell.');
+                            } else {
+                                for (var i = 0; i < jexcel.current.headers.length; i++) {
+                                    jexcel.current.headers[i].classList.remove('dragging-left');
+                                    jexcel.current.headers[i].classList.remove('dragging-right');
+                                }
+    
+                                if (jexcel.current.dragging.column == columnId) {
+                                    jexcel.current.dragging.destination = parseInt(columnId);
+                                } else {
+                                    if (e.target.clientWidth / 2 > e.offsetX) {
+                                        if (jexcel.current.dragging.column < columnId) {
+                                            jexcel.current.dragging.destination = parseInt(columnId) - 1;
+                                        } else {
+                                            jexcel.current.dragging.destination = parseInt(columnId);
+                                        }
+                                        jexcel.current.headers[columnId].classList.add('dragging-left');
+                                    } else {
+                                        if (jexcel.current.dragging.column < columnId) {
+                                            jexcel.current.dragging.destination = parseInt(columnId);
+                                        } else {
+                                            jexcel.current.dragging.destination = parseInt(columnId) + 1;
+                                        }
+                                        jexcel.current.headers[columnId].classList.add('dragging-right');
+                                    }
+                                }
+                            }
+                        }
+                    } else {
+                        if (rowId) {
+                            if (jexcel.current.isRowMerged(rowId).length) {
+                                console.error('JEXCEL: This row is part of a merged cell.');
+                            } else {
+                                var target = (e.target.clientHeight / 2 > e.offsetY) ? e.target.parentNode.nextSibling : e.target.parentNode;
+                                e.target.parentNode.parentNode.insertBefore(jexcel.current.dragging.element, target);
+                            }
+                        }
+                    }
+                } else if (jexcel.current.resizing) {
+                } else {
+                    // Header found
+                    if (jexcelTable[1] == 1) {
+                        if (jexcel.current.selectedHeader) {
+                            var columnId = e.target.getAttribute('data-x');
+                            var o = jexcel.current.selectedHeader;
+                            var d = columnId;
+                            // Update selection
+                            jexcel.current.updateSelectionFromCoords(o, 0, d, jexcel.current.options.data.length - 1);
+                        }
+                    }
+    
+                    // Body found
+                    if (jexcelTable[1] == 2) {
+                        if (e.target.classList.contains('jexcel_row')) {
+                            if (jexcel.current.selectedRow) {
+                                var o = jexcel.current.selectedRow;
+                                var d = rowId;
+                                // Update selection
+                                jexcel.current.updateSelectionFromCoords(0, o, jexcel.current.options.data[0].length - 1, d);
+                            }
+                        } else {
+                            // Do not select edtion is in progress
+                            if (! jexcel.current.edition) {
+                                if (columnId && rowId) {
+                                    if (jexcel.current.selectedCorner) {
+                                        jexcel.current.updateCopySelection(columnId, rowId);
+                                    } else {
+                                        if (jexcel.current.selectedCell) {
+                                            jexcel.current.updateSelectionFromCoords(jexcel.current.selectedCell[0], jexcel.current.selectedCell[1], columnId, rowId);
+                                        }
+                                    }
+                                }
+                            }
+                        }
+                    }
+                }
+            }
+        }
+    
+        // Clear any time control
+        if (jexcel.timeControl) {
+            clearTimeout(jexcel.timeControl);
+            jexcel.timeControl = null;
+        }
+    }
+    
+    /**
+     * Double click event handler: controls the double click in the corner, cell edition or column re-ordering.
+     */
+    jexcel.doubleClickControls = function(e) {
+        // Jexcel is selected
+        if (jexcel.current) {
+            // Corner action
+            if (e.target.classList.contains('jexcel_corner')) {
+                // Any selected cells
+                if (jexcel.current.highlighted.length > 0) {
+                    // Copy from this
+                    var x1 = jexcel.current.highlighted[0].getAttribute('data-x');
+                    var y1 = parseInt(jexcel.current.highlighted[jexcel.current.highlighted.length - 1].getAttribute('data-y')) + 1;
+                    // Until this
+                    var x2 = jexcel.current.highlighted[jexcel.current.highlighted.length - 1].getAttribute('data-x');
+                    var y2 = jexcel.current.records.length - 1
+                    // Execute copy
+                    jexcel.current.copyData(jexcel.current.records[y1][x1], jexcel.current.records[y2][x2]);
+                }
+            } else if (e.target.classList.contains('jexcel_column_filter')) {
+                // Column
+                var columnId = e.target.getAttribute('data-x');
+                // Open filter
+                jexcel.current.openFilter(columnId);
+                
+            } else {
+                // Get table
+                var jexcelTable = jexcel.getElement(e.target);
+    
+                // Double click over header
+                if (jexcelTable[1] == 1 && jexcel.current.options.columnSorting == true) {
+                    // Check valid column header coords
+                    var columnId = e.target.getAttribute('data-x');
+                    if (columnId) {
+                        jexcel.current.orderBy(columnId);
+                    }
+                }
+    
+                // Double click over body
+                if (jexcelTable[1] == 2 && jexcel.current.options.editable == true) {
+                    if (! jexcel.current.edition) {
+                        var getCellCoords = function(element) {
+                            if (element.parentNode) {
+                                var x = element.getAttribute('data-x');
+                                var y = element.getAttribute('data-y');
+                                if (x && y) {
+                                    return element;
+                                } else {
+                                    return getCellCoords(element.parentNode);
+                                }
+                            }
+                        }
+                        var cell = getCellCoords(e.target);
+                        if (cell && cell.classList.contains('highlight')) {
+                            jexcel.current.openEditor(cell);
+                        }
+                    }
+                }
+            }
+        }
+    }
+    
+    jexcel.copyControls = function(e) {
+        if (jexcel.current && jexcel.copyControls.enabled) {
+            if (! jexcel.current.edition) {
+                jexcel.current.copy(true);
+            }
+        }
+    }
+    
+    jexcel.cutControls = function(e) {
+        if (jexcel.current) {
+            if (! jexcel.current.edition) {
+                jexcel.current.copy(true);
+                if (jexcel.current.options.editable == true) {
+                    jexcel.current.setValue(jexcel.current.highlighted, '');
+                }
+            }
+        }
+    }
+    
+    jexcel.pasteControls = function(e) {
+        if (jexcel.current && jexcel.current.selectedCell) {
+            if (! jexcel.current.edition) {
+                if (jexcel.current.options.editable == true) {
+                    if (e && e.clipboardData) {
+                        jexcel.current.paste(jexcel.current.selectedCell[0], jexcel.current.selectedCell[1], e.clipboardData.getData('text'));
+                        e.preventDefault();
+                    } else if (window.clipboardData) {
+                        jexcel.current.paste(jexcel.current.selectedCell[0], jexcel.current.selectedCell[1], window.clipboardData.getData('text'));
+                    }
+                }
+            }
+        }
+    }
+    
+    jexcel.contextMenuControls = function(e) {
+        e = e || window.event;
+        if ("buttons" in e) {
+            var mouseButton = e.buttons;
+        } else {
+            var mouseButton = e.which || e.button;
+        }
+    
+        if (jexcel.current) {
+            if (jexcel.current.edition) {
+                e.preventDefault();
+            } else if (jexcel.current.options.contextMenu) {
+                jexcel.current.contextMenu.contextmenu.close();
+    
+                if (jexcel.current) {
+                    var x = e.target.getAttribute('data-x');
+                    var y = e.target.getAttribute('data-y');
+    
+                    if (x || y) {
+                        // Table found
+                        var items = jexcel.current.options.contextMenu(jexcel.current, x, y, e);
+                        // The id is depending on header and body
+                        jexcel.current.contextMenu.contextmenu.open(e, items);
+                        // Avoid the real one
+                        e.preventDefault();
+                    }
+                }
+            }
+        }
+    }
+
+    jexcel.touchStartControls = function(e) {
+        var jexcelTable = jexcel.getElement(e.target);
+    
+        if (jexcelTable[0]) {
+            if (jexcel.current != jexcelTable[0].jexcel) {
+                if (jexcel.current) {
+                    jexcel.current.resetSelection();
+                }
+                jexcel.current = jexcelTable[0].jexcel;
+            }
+        } else {
+            if (jexcel.current) {
+                jexcel.current.resetSelection();
+                jexcel.current = null;
+            }
+        }
+    
+        if (jexcel.current) {
+            if (! jexcel.current.edition) {
+                var columnId = e.target.getAttribute('data-x');
+                var rowId = e.target.getAttribute('data-y');
+    
+                if (columnId && rowId) {
+                    jexcel.current.updateSelectionFromCoords(columnId, rowId);
+    
+                    jexcel.timeControl = setTimeout(function() {
+                        // Keep temporary reference to the element
+                        if (jexcel.current.options.columns[columnId].type == 'color') {
+                            jexcel.tmpElement = null;
+                        } else {
+                            jexcel.tmpElement = e.target;
+                        }
+                        jexcel.current.openEditor(e.target, false, e);
+                    }, 500);
+                }
+            }
+        }
+    }
+    
+    jexcel.touchEndControls = function(e) {
+        // Clear any time control
+        if (jexcel.timeControl) {
+            clearTimeout(jexcel.timeControl);
+            jexcel.timeControl = null;
+            // Element
+            if (jexcel.tmpElement && jexcel.tmpElement.children[0].tagName == 'INPUT') {
+                jexcel.tmpElement.children[0].focus();
+            }
+            jexcel.tmpElement = null;
+        }
+    }
+    
+    /**
+     * Jexcel extensions
+     */
+    
+    jexcel.tabs = function(tabs, result) {
+        var instances = [];
+        // Create tab container
+        if (! tabs.classList.contains('jexcel_tabs')) {
+            tabs.innerHTML = '';
+            tabs.classList.add('jexcel_tabs')
+            tabs.jexcel = [];
+
+            var div = document.createElement('div');
+            var headers = tabs.appendChild(div);
+            var div = document.createElement('div');
+            var content = tabs.appendChild(div);
+        } else {
+            var headers = tabs.children[0];
+            var content = tabs.children[1];
+        }
+
+        var spreadsheet = []
+        var link = [];
+        for (var i = 0; i < result.length; i++) {
+            // Spreadsheet container
+            spreadsheet[i] = document.createElement('div');
+            spreadsheet[i].classList.add('jexcel_tab');
+            var worksheet = jexcel(spreadsheet[i], result[i]);
+            content.appendChild(spreadsheet[i]);
+            instances[i] = tabs.jexcel.push(worksheet);
+
+            // Tab link
+            link[i] = document.createElement('div');
+            link[i].classList.add('jexcel_tab_link');
+            link[i].setAttribute('data-spreadsheet', tabs.jexcel.length-1);
+            link[i].innerHTML = result[i].sheetName;
+            link[i].onclick = function() {
+                for (var j = 0; j < headers.children.length; j++) {
+                    headers.children[j].classList.remove('selected');
+                    content.children[j].style.display = 'none';
+                }
+                var i = this.getAttribute('data-spreadsheet');
+                content.children[i].style.display = 'block';
+                headers.children[i].classList.add('selected')
+            }
+            headers.appendChild(link[i]);
+        }
+
+        // First tab
+        for (var j = 0; j < headers.children.length; j++) {
+            headers.children[j].classList.remove('selected');
+            content.children[j].style.display = 'none';
+        }
+        headers.children[headers.children.length - 1].classList.add('selected');
+        content.children[headers.children.length - 1].style.display = 'block';
+
+        return instances;
+    }
+
+    // Compability to older versions
+    jexcel.createTabs = jexcel.tabs;
+
+    jexcel.fromSpreadsheet = function(file, __callback) {
+        var convert = function(workbook) {
+            var spreadsheets = [];
+            workbook.SheetNames.forEach(function(sheetName) {
+                var spreadsheet = {};
+                spreadsheet.rows = [];
+                spreadsheet.columns = [];
+                spreadsheet.data = [];
+                spreadsheet.style = {};
+                spreadsheet.sheetName = sheetName;
+    
+                // Column widths
+                var temp = workbook.Sheets[sheetName]['!cols'];
+                if (temp && temp.length) {
+                    for (var i = 0; i < temp.length; i++) {
+                        spreadsheet.columns[i] = {};
+                        if (temp[i] && temp[i].wpx) {
+                            spreadsheet.columns[i].width = temp[i].wpx + 'px';
+                        }
+                     }
+                }
+                // Rows heights
+                var temp = workbook.Sheets[sheetName]['!rows'];
+                if (temp && temp.length) {
+                    for (var i = 0; i < temp.length; i++) {
+                        if (temp[i] && temp[i].hpx) {
+                            spreadsheet.rows[i] = {};
+                            spreadsheet.rows[i].height = temp[i].hpx + 'px';
+                        }
+                    }
+                }
+                // Merge cells
+                var temp = workbook.Sheets[sheetName]['!merges'];
+                if (temp && temp.length > 0) {
+                    spreadsheet.mergeCells = [];
+                    for (var i = 0; i < temp.length; i++) {
+                        var x1 = temp[i].s.c;
+                        var y1 = temp[i].s.r;
+                        var x2 = temp[i].e.c;
+                        var y2 = temp[i].e.r;
+                        var key = jexcel.getColumnNameFromId([x1,y1]);
+                        spreadsheet.mergeCells[key] = [ x2-x1+1, y2-y1+1 ];
+                    }
+                }
+                // Data container
+                var max_x = 0;
+                var max_y = 0;
+                var temp = Object.keys(workbook.Sheets[sheetName]);
+                for (var i = 0; i < temp.length; i++) {
+                    if (temp[i].substr(0,1) != '!') {
+                        var cell = workbook.Sheets[sheetName][temp[i]];
+                        var info = jexcel.getIdFromColumnName(temp[i], true);
+                        if (! spreadsheet.data[info[1]]) {
+                            spreadsheet.data[info[1]] = [];
+                        }
+                        spreadsheet.data[info[1]][info[0]] = cell.f ? '=' + cell.f : cell.w;
+                        if (max_x < info[0]) {
+                            max_x = info[0];
+                        }
+                        if (max_y < info[1]) {
+                            max_y = info[1];
+                        }
+                        // Style
+                        if (cell.style && Object.keys(cell.style).length > 0) {
+                            spreadsheet.style[temp[i]] = cell.style;
+                        }
+                        if (cell.s && cell.s.fgColor) {
+                            if (spreadsheet.style[temp[i]]) {
+                                spreadsheet.style[temp[i]] += ';';
+                            }
+                            spreadsheet.style[temp[i]] += 'background-color:#' + cell.s.fgColor.rgb;
+                        }
+                    }
+                }
+                var numColumns = spreadsheet.columns;
+                for (var j = 0; j <= max_y; j++) {
+                    for (var i = 0; i <= max_x; i++) {
+                        if (! spreadsheet.data[j]) {
+                            spreadsheet.data[j] = [];
+                        }
+                        if (! spreadsheet.data[j][i]) {
+                            if (numColumns < i) {
+                                spreadsheet.data[j][i] = '';
+                            }
+                        }
+                    }
+                }
+                spreadsheets.push(spreadsheet);
+            });
+    
+            return spreadsheets;
+        }
+    
+        var oReq;
+        oReq = new XMLHttpRequest();
+        oReq.open("GET", file, true);
+    
+        if(typeof Uint8Array !== 'undefined') {
+            oReq.responseType = "arraybuffer";
+            oReq.onload = function(e) {
+                var arraybuffer = oReq.response;
+                var data = new Uint8Array(arraybuffer);
+                var wb = XLSX.read(data, {type:"array", cellFormula:true, cellStyles:true });
+                __callback(convert(wb))
+            };
+        } else {
+            oReq.setRequestHeader("Accept-Charset", "x-user-defined");  
+            oReq.onreadystatechange = function() { if(oReq.readyState == 4 && oReq.status == 200) {
+                var ff = convertResponseBodyToText(oReq.responseBody);
+                var wb = XLSX.read(ff, {type:"binary", cellFormula:true, cellStyles:true });
+                __callback(convert(wb))
+            }};
+        }
+    
+        oReq.send();
+    }
+    
+    /**
+     * Valid international letter
+     */
+    
+    jexcel.validLetter = function (text) {
+        var regex = /([\u0041-\u005A\u0061-\u007A\u00AA\u00B5\u00BA\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02C1\u02C6-\u02D1\u02E0-\u02E4\u02EC\u02EE\u0370-\u0374\u0376\u0377\u037A-\u037D\u0386\u0388-\u038A\u038C\u038E-\u03A1\u03A3-\u03F5\u03F7-\u0481\u048A-\u0527\u0531-\u0556\u0559\u0561-\u0587\u05D0-\u05EA\u05F0-\u05F2\u0620-\u064A\u066E\u066F\u0671-\u06D3\u06D5\u06E5\u06E6\u06EE\u06EF\u06FA-\u06FC\u06FF\u0710\u0712-\u072F\u074D-\u07A5\u07B1\u07CA-\u07EA\u07F4\u07F5\u07FA\u0800-\u0815\u081A\u0824\u0828\u0840-\u0858\u08A0\u08A2-\u08AC\u0904-\u0939\u093D\u0950\u0958-\u0961\u0971-\u0977\u0979-\u097F\u0985-\u098C\u098F\u0990\u0993-\u09A8\u09AA-\u09B0\u09B2\u09B6-\u09B9\u09BD\u09CE\u09DC\u09DD\u09DF-\u09E1\u09F0\u09F1\u0A05-\u0A0A\u0A0F\u0A10\u0A13-\u0A28\u0A2A-\u0A30\u0A32\u0A33\u0A35\u0A36\u0A38\u0A39\u0A59-\u0A5C\u0A5E\u0A72-\u0A74\u0A85-\u0A8D\u0A8F-\u0A91\u0A93-\u0AA8\u0AAA-\u0AB0\u0AB2\u0AB3\u0AB5-\u0AB9\u0ABD\u0AD0\u0AE0\u0AE1\u0B05-\u0B0C\u0B0F\u0B10\u0B13-\u0B28\u0B2A-\u0B30\u0B32\u0B33\u0B35-\u0B39\u0B3D\u0B5C\u0B5D\u0B5F-\u0B61\u0B71\u0B83\u0B85-\u0B8A\u0B8E-\u0B90\u0B92-\u0B95\u0B99\u0B9A\u0B9C\u0B9E\u0B9F\u0BA3\u0BA4\u0BA8-\u0BAA\u0BAE-\u0BB9\u0BD0\u0C05-\u0C0C\u0C0E-\u0C10\u0C12-\u0C28\u0C2A-\u0C33\u0C35-\u0C39\u0C3D\u0C58\u0C59\u0C60\u0C61\u0C85-\u0C8C\u0C8E-\u0C90\u0C92-\u0CA8\u0CAA-\u0CB3\u0CB5-\u0CB9\u0CBD\u0CDE\u0CE0\u0CE1\u0CF1\u0CF2\u0D05-\u0D0C\u0D0E-\u0D10\u0D12-\u0D3A\u0D3D\u0D4E\u0D60\u0D61\u0D7A-\u0D7F\u0D85-\u0D96\u0D9A-\u0DB1\u0DB3-\u0DBB\u0DBD\u0DC0-\u0DC6\u0E01-\u0E30\u0E32\u0E33\u0E40-\u0E46\u0E81\u0E82\u0E84\u0E87\u0E88\u0E8A\u0E8D\u0E94-\u0E97\u0E99-\u0E9F\u0EA1-\u0EA3\u0EA5\u0EA7\u0EAA\u0EAB\u0EAD-\u0EB0\u0EB2\u0EB3\u0EBD\u0EC0-\u0EC4\u0EC6\u0EDC-\u0EDF\u0F00\u0F40-\u0F47\u0F49-\u0F6C\u0F88-\u0F8C\u1000-\u102A\u103F\u1050-\u1055\u105A-\u105D\u1061\u1065\u1066\u106E-\u1070\u1075-\u1081\u108E\u10A0-\u10C5\u10C7\u10CD\u10D0-\u10FA\u10FC-\u1248\u124A-\u124D\u1250-\u1256\u1258\u125A-\u125D\u1260-\u1288\u128A-\u128D\u1290-\u12B0\u12B2-\u12B5\u12B8-\u12BE\u12C0\u12C2-\u12C5\u12C8-\u12D6\u12D8-\u1310\u1312-\u1315\u1318-\u135A\u1380-\u138F\u13A0-\u13F4\u1401-\u166C\u166F-\u167F\u1681-\u169A\u16A0-\u16EA\u1700-\u170C\u170E-\u1711\u1720-\u1731\u1740-\u1751\u1760-\u176C\u176E-\u1770\u1780-\u17B3\u17D7\u17DC\u1820-\u1877\u1880-\u18A8\u18AA\u18B0-\u18F5\u1900-\u191C\u1950-\u196D\u1970-\u1974\u1980-\u19AB\u19C1-\u19C7\u1A00-\u1A16\u1A20-\u1A54\u1AA7\u1B05-\u1B33\u1B45-\u1B4B\u1B83-\u1BA0\u1BAE\u1BAF\u1BBA-\u1BE5\u1C00-\u1C23\u1C4D-\u1C4F\u1C5A-\u1C7D\u1CE9-\u1CEC\u1CEE-\u1CF1\u1CF5\u1CF6\u1D00-\u1DBF\u1E00-\u1F15\u1F18-\u1F1D\u1F20-\u1F45\u1F48-\u1F4D\u1F50-\u1F57\u1F59\u1F5B\u1F5D\u1F5F-\u1F7D\u1F80-\u1FB4\u1FB6-\u1FBC\u1FBE\u1FC2-\u1FC4\u1FC6-\u1FCC\u1FD0-\u1FD3\u1FD6-\u1FDB\u1FE0-\u1FEC\u1FF2-\u1FF4\u1FF6-\u1FFC\u2071\u207F\u2090-\u209C\u2102\u2107\u210A-\u2113\u2115\u2119-\u211D\u2124\u2126\u2128\u212A-\u212D\u212F-\u2139\u213C-\u213F\u2145-\u2149\u214E\u2183\u2184\u2C00-\u2C2E\u2C30-\u2C5E\u2C60-\u2CE4\u2CEB-\u2CEE\u2CF2\u2CF3\u2D00-\u2D25\u2D27\u2D2D\u2D30-\u2D67\u2D6F\u2D80-\u2D96\u2DA0-\u2DA6\u2DA8-\u2DAE\u2DB0-\u2DB6\u2DB8-\u2DBE\u2DC0-\u2DC6\u2DC8-\u2DCE\u2DD0-\u2DD6\u2DD8-\u2DDE\u2E2F\u3005\u3006\u3031-\u3035\u303B\u303C\u3041-\u3096\u309D-\u309F\u30A1-\u30FA\u30FC-\u30FF\u3105-\u312D\u3131-\u318E\u31A0-\u31BA\u31F0-\u31FF\u3400-\u4DB5\u4E00-\u9FCC\uA000-\uA48C\uA4D0-\uA4FD\uA500-\uA60C\uA610-\uA61F\uA62A\uA62B\uA640-\uA66E\uA67F-\uA697\uA6A0-\uA6E5\uA717-\uA71F\uA722-\uA788\uA78B-\uA78E\uA790-\uA793\uA7A0-\uA7AA\uA7F8-\uA801\uA803-\uA805\uA807-\uA80A\uA80C-\uA822\uA840-\uA873\uA882-\uA8B3\uA8F2-\uA8F7\uA8FB\uA90A-\uA925\uA930-\uA946\uA960-\uA97C\uA984-\uA9B2\uA9CF\uAA00-\uAA28\uAA40-\uAA42\uAA44-\uAA4B\uAA60-\uAA76\uAA7A\uAA80-\uAAAF\uAAB1\uAAB5\uAAB6\uAAB9-\uAABD\uAAC0\uAAC2\uAADB-\uAADD\uAAE0-\uAAEA\uAAF2-\uAAF4\uAB01-\uAB06\uAB09-\uAB0E\uAB11-\uAB16\uAB20-\uAB26\uAB28-\uAB2E\uABC0-\uABE2\uAC00-\uD7A3\uD7B0-\uD7C6\uD7CB-\uD7FB\uF900-\uFA6D\uFA70-\uFAD9\uFB00-\uFB06\uFB13-\uFB17\uFB1D\uFB1F-\uFB28\uFB2A-\uFB36\uFB38-\uFB3C\uFB3E\uFB40\uFB41\uFB43\uFB44\uFB46-\uFBB1\uFBD3-\uFD3D\uFD50-\uFD8F\uFD92-\uFDC7\uFDF0-\uFDFB\uFE70-\uFE74\uFE76-\uFEFC\uFF21-\uFF3A\uFF41-\uFF5A\uFF66-\uFFBE\uFFC2-\uFFC7\uFFCA-\uFFCF\uFFD2-\uFFD7\uFFDA-\uFFDC-\u0400-\u04FF']+)/g;
+        return text.match(regex) ? 1 : 0;
+    }
+    
+    /**
+     * Helper injectArray
+     */
+    jexcel.injectArray = function(o, idx, arr) {
+        return o.slice(0, idx).concat(arr).concat(o.slice(idx));
+    }
+    
+    /**
+     * Get letter based on a number
+     * 
+     * @param integer i
+     * @return string letter
+     */
+    jexcel.getColumnName = function(i) {
+        var letter = '';
+        if (i > 701) {
+            letter += String.fromCharCode(64 + parseInt(i / 676));
+            letter += String.fromCharCode(64 + parseInt((i % 676) / 26));
+        } else if (i > 25) {
+            letter += String.fromCharCode(64 + parseInt(i / 26));
+        }
+        letter += String.fromCharCode(65 + (i % 26));
+    
+        return letter;
+    }
+    
+    /**
+     * Convert excel like column to jexcel id
+     * 
+     * @param string id
+     * @return string id
+     */
+    jexcel.getIdFromColumnName = function (id, arr) {
+        // Get the letters
+        var t = /^[a-zA-Z]+/.exec(id);
+    
+        if (t) {
+            // Base 26 calculation
+            var code = 0;
+            for (var i = 0; i < t[0].length; i++) {
+                code += parseInt(t[0].charCodeAt(i) - 64) * Math.pow(26, (t[0].length - 1 - i));
+            }
+            code--;
+            // Make sure jexcel starts on zero
+            if (code < 0) {
+                code = 0;
+            }
+    
+            // Number
+            var number = parseInt(/[0-9]+$/.exec(id));
+            if (number > 0) {
+                number--;
+            }
+    
+            if (arr == true) {
+                id = [ code, number ];
+            } else {
+                id = code + '-' + number;
+            }
+        }
+    
+        return id;
+    }
+    
+    /**
+     * Convert jexcel id to excel like column name
+     * 
+     * @param string id
+     * @return string id
+     */
+    jexcel.getColumnNameFromId = function (cellId) {
+        if (! Array.isArray(cellId)) {
+            cellId = cellId.split('-');
+        }
+    
+        return jexcel.getColumnName(parseInt(cellId[0])) + (parseInt(cellId[1]) + 1);
+    }
+    
+    /**
+     * Verify element inside jexcel table
+     * 
+     * @param string id
+     * @return string id
+     */
+    jexcel.getElement = function(element) {
+        var jexcelSection = 0;
+        var jexcelElement = 0;
+    
+        function path (element) {
+            if (element.className) {
+                if (element.classList.contains('jexcel_container')) {
+                    jexcelElement = element;
+                }
+            }
+    
+            if (element.tagName == 'THEAD') {
+                jexcelSection = 1;
+            } else if (element.tagName == 'TBODY') {
+                jexcelSection = 2;
+            }
+    
+            if (element.parentNode) {
+                if (! jexcelElement) {
+                    path(element.parentNode);
+                }
+            }
+        }
+    
+        path(element);
+    
+        return [ jexcelElement, jexcelSection ];
+    }
+
+    jexcel.doubleDigitFormat = function(v) {
+        v = ''+v;
+        if (v.length == 1) {
+            v = '0'+v;
+        }
+        return v;
+    }
+
+    jexcel.createFromTable = function(el, options) {
+        if (el.tagName != 'TABLE') {
+            console.log('Element is not a table');
+        } else {
+            // Configuration
+            if (! options) {
+                options = {};
+            }
+            options.columns = [];
+            options.data = [];
+
+            // Colgroup
+            var colgroup = el.querySelectorAll('colgroup > col');
+            if (colgroup.length) {
+                // Get column width
+                for (var i = 0; i < colgroup.length; i++) {
+                    var width = colgroup[i].style.width;
+                    if (! width) {
+                        var width = colgroup[i].getAttribute('width');
+                    }
+                    // Set column width
+                    if (width) {
+                        if (! options.columns[i]) {
+                            options.columns[i] = {}
+                        }
+                        options.columns[i].width = width;
+                    }
+                }
+            }
+
+            // Parse header
+            var parseHeader = function(header) {
+                // Get width information
+                var info = header.getBoundingClientRect();
+                var width = info.width > 50 ? info.width : 50;
+
+                // Create column option
+                if (! options.columns[i]) {
+                    options.columns[i] = {};
+                } 
+                if (header.getAttribute('data-celltype')) {
+                    options.columns[i].type = header.getAttribute('data-celltype');
+                } else {
+                    options.columns[i].type = 'text';
+                }
+                options.columns[i].width = width + 'px';
+                options.columns[i].title = header.innerText;
+                options.columns[i].align = header.style.textAlign || 'center';
+            }
+
+            // Headers
+            var headers = el.querySelectorAll('thead > tr');
+            if (headers.length) {
+                // Get the last row in the thead
+                headers = headers[headers.length-1].children;
+                // Go though the headers
+                for (var i = 0; i < headers.length; i++) {
+                    parseHeader(headers[i]);
+                }
+            }
+
+            // Content
+            var rowNumber = 0;
+            var mergeCells = {};
+            var rows = {};
+            var style = {};
+
+            var content = el.querySelectorAll('table > tr, tbody tr');
+            for (var j = 0; j < content.length; j++) {
+                options.data[rowNumber] = [];
+                if (options.parseTableFirstRowAsHeader == true && j == 0) {
+                    for (var i = 0; i < content[j].children.length; i++) {
+                        parseHeader(content[j].children[i]);
+                    }
+                } else {
+                    for (var i = 0; i < content[j].children.length; i++) {
+                        // WickedGrid formula compatibility
+                        var value = content[j].children[i].getAttribute('formula');
+                        if (!value) {
+                          value = content[j].children[i].getAttribute('data-formula');
+                        }
+                        if (value) {
+                            if (value.substr(0,1) != '=') {
+                                value = '=' + value;
+                            }
+                        } else {
+                            var value = content[j].children[i].innerText;
+                        }
+                        options.data[rowNumber].push(value);
+
+                        // Key
+                        var cellName = jexcel.getColumnNameFromId([ i, j ]);
+
+                        // Merged cells
+                        var mergedColspan = parseInt(content[j].children[i].getAttribute('colspan')) || 0;
+                        var mergedRowspan = parseInt(content[j].children[i].getAttribute('rowspan')) || 0;
+                        if (mergedColspan || mergedRowspan) {
+                            mergeCells[cellName] = [ mergedColspan || 1, mergedRowspan || 1 ];
+                        }
+
+                        // Avoid problems with hidden cells
+                        if (s = content[j].children[i].style && content[j].children[i].style.display == 'none') {
+                            content[j].children[i].style.display = '';
+                        }
+                        // Get style
+                        var s = content[j].children[i].getAttribute('style');
+                        if (s) {
+                            style[cellName] = s;
+                        }
+                        // Bold
+                        if (content[j].children[i].classList.contains('styleBold') || content[j].children[i].classList.contains('style-bold')) {
+                            if (style[cellName]) {
+                                style[cellName] += '; font-weight:bold;';
+                            } else {
+                                style[cellName] = 'font-weight:bold;';
+                            }
+                        }
+                        // Italic
+                        if (content[j].children[i].classList.contains('styleItalics') || content[j].children[i].classList.contains('style-italics')) {
+                            if (style[cellName]) {
+                                style[cellName] += '; font-style: italic;';
+                            } else {
+                                style[cellName] = 'font-style: italic;';
+                            }
+                        }
+                        // Underlined
+                        if (content[j].children[i].classList.contains('styleUnderline') || content[j].children[i].classList.contains('style-underline')) {
+                            if (style[cellName]) {
+                                style[cellName] += '; text-decoration: underline;';
+                            } else {
+                                style[cellName] = 'text-decoration: underline;';
+                            }
+                        }
+                        // Line through
+                        if (content[j].children[i].classList.contains('style-line-through')) {
+                            if (style[cellName]) {
+                                style[cellName] += '; text-decoration: line-through;';
+                            } else {
+                                style[cellName] = 'text-decoration: line-through;';
+                            }
+                        }
+                        // Align left
+                        if (content[j].children[i].classList.contains('styleLeft')) {
+                            if (style[cellName]) {
+                                style[cellName] += '; text-align: left;';
+                            } else {
+                                style[cellName] = 'text-align: left;';
+                            }
+                        }
+                        // Align right
+                        if (content[j].children[i].classList.contains('styleRight')) {
+                            if (style[cellName]) {
+                                style[cellName] += '; text-align: right;';
+                            } else {
+                                style[cellName] = 'text-align: right;';
+                            }
+                        }
+                    }
+
+                    // Row Height
+                    if (content[j].style && content[j].style.height) {
+                        rows[j] = { height: content[j].style.height };
+                    }
+
+                    // Index
+                    rowNumber++;
+                }
+            }
+
+            // Style
+            if (Object.keys(style).length > 0) {
+                options.style = style;
+            }
+            // Merged
+            if (Object.keys(mergeCells).length > 0) {
+                options.mergeCells = mergeCells;
+            }
+            // Row height
+            if (Object.keys(rows).length > 0) {
+                options.rows = rows;
+            }
+
+            // TODO: data-hiddencolumns="3,4"
+            
+            // I guess in terms the better column type
+            if (options.parseTableAutoCellType == true) {
+                var pattern = [];
+                for (var i = 0; i < options.columns.length; i++) {
+                    var test = true;
+                    var testCalendar = true;
+                    pattern[i] = [];
+                    for (var j = 0; j < options.data.length; j++) {
+                        var value = options.data[j][i];
+                        if (! pattern[i][value]) {
+                            pattern[i][value] = 0;
+                        }
+                        pattern[i][value]++;
+                        if (value.length > 25) {
+                            test = false;
+                        }
+                        if (value.length == 10) {
+                            if (! (value.substr(4,1) == '-' && value.substr(7,1) == '-')) {
+                                testCalendar = false;
+                            }
+                        } else {
+                            testCalendar = false;
+                        }
+                    }
+
+                    var keys = Object.keys(pattern[i]).length;
+                    if (testCalendar) {
+                        options.columns[i].type = 'calendar';
+                    } else if (test == true && keys > 1 && keys <= parseInt(options.data.length * 0.1)) {
+                        options.columns[i].type = 'dropdown';
+                        options.columns[i].source = Object.keys(pattern[i]);
+                    }
+                }
+            }
+
+            return options;
+        }
+    }
+
+    /**
+     * Jquery Support
+     */
+    if (typeof(jQuery) != 'undefined') {
+        (function($){
+            $.fn.jexcel = function(method) {
+                var spreadsheetContainer = $(this).get(0);
+                if (! spreadsheetContainer.jexcel) {
+                    return jexcel($(this).get(0), arguments[0]);
+                } else {
+                    return spreadsheetContainer.jexcel[method].apply(this, Array.prototype.slice.call( arguments, 1 ));
+                }
+            };
+    
+        })(jQuery);
+    }
+    
+    
+    // Based on sutoiku work (https://github.com/sutoiku)
+    
+    var error = (function() {
+        var exports = {};
+    
+        exports.nil = new Error('#NULL!');
+        exports.div0 = new Error('#DIV/0!');
+        exports.value = new Error('#VALUE!');
+        exports.ref = new Error('#REF!');
+        exports.name = new Error('#NAME?');
+        exports.num = new Error('#NUM!');
+        exports.na = new Error('#N/A');
+        exports.error = new Error('#ERROR!');
+        exports.data = new Error('#GETTING_DATA');
+    
+        return exports;
+    })();
+    
+    var utils = (function() {
+        var exports = {};
+    
+        exports.flattenShallow = function(array) {
+            if (!array || !array.reduce) {
+                return array;
+            }
+    
+            return array.reduce(function(a, b) {
+                var aIsArray = Array.isArray(a);
+                var bIsArray = Array.isArray(b);
+    
+                if (aIsArray && bIsArray) {
+                    return a.concat(b);
+                }
+                if (aIsArray) {
+                    a.push(b);
+    
+                    return a;
+                }
+                if (bIsArray) {
+                    return [ a ].concat(b);
+                }
+    
+                return [ a, b ];
+            });
+        };
+    
+        exports.isFlat = function(array) {
+            if (!array) {
+                return false;
+            }
+    
+            for (var i = 0; i < array.length; ++i) {
+                if (Array.isArray(array[i])) {
+                    return false;
+                }
+            }
+    
+            return true;
+        };
+    
+        exports.flatten = function() {
+            var result = exports.argsToArray.apply(null, arguments);
+    
+            while (!exports.isFlat(result)) {
+                result = exports.flattenShallow(result);
+            }
+    
+            return result;
+        };
+    
+        exports.argsToArray = function(args) {
+            var result = [];
+    
+            exports.arrayEach(args, function(value) {
+                result.push(value);
+            });
+    
+            return result;
+        };
+    
+        exports.numbers = function() {
+            var possibleNumbers = this.flatten.apply(null, arguments);
+            return possibleNumbers.filter(function(el) {
+                return typeof el === 'number';
+            });
+        };
+    
+        exports.cleanFloat = function(number) {
+            var power = 1e14;
+            return Math.round(number * power) / power;
+        };
+    
+        exports.parseBool = function(bool) {
+            if (typeof bool === 'boolean') {
+                return bool;
+            }
+    
+            if (bool instanceof Error) {
+                return bool;
+            }
+    
+            if (typeof bool === 'number') {
+                return bool !== 0;
+            }
+    
+            if (typeof bool === 'string') {
+                var up = bool.toUpperCase();
+                if (up === 'TRUE') {
+                    return true;
+                }
+    
+                if (up === 'FALSE') {
+                    return false;
+                }
+            }
+    
+            if (bool instanceof Date && !isNaN(bool)) {
+                return true;
+            }
+    
+            return error.value;
+        };
+    
+        exports.parseNumber = function(string) {
+            if (string === undefined || string === '') {
+                return error.value;
+            }
+            if (!isNaN(string)) {
+                return parseFloat(string);
+            }
+    
+            return error.value;
+        };
+    
+        exports.parseNumberArray = function(arr) {
+            var len;
+    
+            if (!arr || (len = arr.length) === 0) {
+                return error.value;
+            }
+    
+            var parsed;
+    
+            while (len--) {
+                parsed = exports.parseNumber(arr[len]);
+                if (parsed === error.value) {
+                    return parsed;
+                }
+                arr[len] = parsed;
+            }
+    
+            return arr;
+        };
+    
+        exports.parseMatrix = function(matrix) {
+            var n;
+    
+            if (!matrix || (n = matrix.length) === 0) {
+                return error.value;
+            }
+            var pnarr;
+    
+            for (var i = 0; i < matrix.length; i++) {
+                pnarr = exports.parseNumberArray(matrix[i]);
+                matrix[i] = pnarr;
+    
+                if (pnarr instanceof Error) {
+                    return pnarr;
+                }
+            }
+    
+            return matrix;
+        };
+    
+        var d1900 = new Date(Date.UTC(1900, 0, 1));
+        exports.parseDate = function(date) {
+            if (!isNaN(date)) {
+                if (date instanceof Date) {
+                    return new Date(date);
+                }
+                var d = parseInt(date, 10);
+                if (d < 0) {
+                    return error.num;
+                }
+                if (d <= 60) {
+                    return new Date(d1900.getTime() + (d - 1) * 86400000);
+                }
+                return new Date(d1900.getTime() + (d - 2) * 86400000);
+            }
+            if (typeof date === 'string') {
+                date = new Date(date);
+                if (!isNaN(date)) {
+                    return date;
+                }
+            }
+            return error.value;
+        };
+    
+        exports.parseDateArray = function(arr) {
+            var len = arr.length;
+            var parsed;
+            while (len--) {
+                parsed = this.parseDate(arr[len]);
+                if (parsed === error.value) {
+                    return parsed;
+                }
+                arr[len] = parsed;
+            }
+            return arr;
+        };
+    
+        exports.anyIsError = function() {
+            var n = arguments.length;
+            while (n--) {
+                if (arguments[n] instanceof Error) {
+                    return true;
+                }
+            }
+            return false;
+        };
+    
+        exports.arrayValuesToNumbers = function(arr) {
+            var n = arr.length;
+            var el;
+            while (n--) {
+                el = arr[n];
+                if (typeof el === 'number') {
+                    continue;
+                }
+                if (el === true) {
+                    arr[n] = 1;
+                    continue;
+                }
+                if (el === false) {
+                    arr[n] = 0;
+                    continue;
+                }
+                if (typeof el === 'string') {
+                    var number = this.parseNumber(el);
+                    if (number instanceof Error) {
+                        arr[n] = 0;
+                    } else {
+                        arr[n] = number;
+                    }
+                }
+            }
+            return arr;
+        };
+    
+        exports.rest = function(array, idx) {
+            idx = idx || 1;
+            if (!array || typeof array.slice !== 'function') {
+                return array;
+            }
+            return array.slice(idx);
+        };
+    
+        exports.initial = function(array, idx) {
+            idx = idx || 1;
+            if (!array || typeof array.slice !== 'function') {
+                return array;
+            }
+            return array.slice(0, array.length - idx);
+        };
+    
+        exports.arrayEach = function(array, iteratee) {
+            var index = -1, length = array.length;
+    
+            while (++index < length) {
+                if (iteratee(array[index], index, array) === false) {
+                    break;
+                }
+            }
+    
+            return array;
+        };
+    
+        exports.transpose = function(matrix) {
+            if (!matrix) {
+                return error.value;
+            }
+    
+            return matrix[0].map(function(col, i) {
+                return matrix.map(function(row) {
+                    return row[i];
+                });
+            });
+        };
+    
+        return exports;
+    })();
+    
+    jexcel.methods = {};
+    
+    jexcel.methods.datetime = (function() {
+        var exports = {};
+    
+        var d1900 = new Date(1900, 0, 1);
+        var WEEK_STARTS = [
+            undefined,
+            0,
+            1,
+            undefined,
+            undefined,
+            undefined,
+            undefined,
+            undefined,
+            undefined,
+            undefined,
+            undefined,
+            undefined,
+            1,
+            2,
+            3,
+            4,
+            5,
+            6,
+            0
+        ];
+        var WEEK_TYPES = [
+            [],
+            [1, 2, 3, 4, 5, 6, 7],
+            [7, 1, 2, 3, 4, 5, 6],
+            [6, 0, 1, 2, 3, 4, 5],
+            [],
+            [],
+            [],
+            [],
+            [],
+            [],
+            [],
+            [7, 1, 2, 3, 4, 5, 6],
+            [6, 7, 1, 2, 3, 4, 5],
+            [5, 6, 7, 1, 2, 3, 4],
+            [4, 5, 6, 7, 1, 2, 3],
+            [3, 4, 5, 6, 7, 1, 2],
+            [2, 3, 4, 5, 6, 7, 1],
+            [1, 2, 3, 4, 5, 6, 7]
+        ];
+        var WEEKEND_TYPES = [
+            [],
+            [6, 0],
+            [0, 1],
+            [1, 2],
+            [2, 3],
+            [3, 4],
+            [4, 5],
+            [5, 6],
+            undefined,
+            undefined,
+            undefined, [0, 0],
+            [1, 1],
+            [2, 2],
+            [3, 3],
+            [4, 4],
+            [5, 5],
+            [6, 6]
+        ];
+    
+        exports.DATE = function(year, month, day) {
+            year = utils.parseNumber(year);
+            month = utils.parseNumber(month);
+            day = utils.parseNumber(day);
+            if (utils.anyIsError(year, month, day)) {
+                return error.value;
+            }
+            if (year < 0 || month < 0 || day < 0) {
+                return error.num;
+            }
+            var date = new Date(year, month - 1, day);
+            return date;
+        };
+    
+        exports.DATEVALUE = function(date_text) {
+            if (typeof date_text !== 'string') {
+                return error.value;
+            }
+            var date = Date.parse(date_text);
+            if (isNaN(date)) {
+                return error.value;
+            }
+            if (date <= -2203891200000) {
+                return (date - d1900) / 86400000 + 1;
+            }
+            return (date - d1900) / 86400000 + 2;
+        };
+    
+        exports.DAY = function(serial_number) {
+            var date = utils.parseDate(serial_number);
+            if (date instanceof Error) {
+                return date;
+            }
+            return date.getDate();
+        };
+    
+        exports.DAYS = function(end_date, start_date) {
+            end_date = utils.parseDate(end_date);
+            start_date = utils.parseDate(start_date);
+            if (end_date instanceof Error) {
+                return end_date;
+            }
+            if (start_date instanceof Error) {
+                return start_date;
+            }
+            return serial(end_date) - serial(start_date);
+        };
+    
+        exports.DAYS360 = function(start_date, end_date, method) {
+        };
+    
+        exports.EDATE = function(start_date, months) {
+            start_date = utils.parseDate(start_date);
+            if (start_date instanceof Error) {
+                return start_date;
+            }
+            if (isNaN(months)) {
+                return error.value;
+            }
+            months = parseInt(months, 10);
+            start_date.setMonth(start_date.getMonth() + months);
+            return serial(start_date);
+        };
+    
+        exports.EOMONTH = function(start_date, months) {
+            start_date = utils.parseDate(start_date);
+            if (start_date instanceof Error) {
+                return start_date;
+            }
+            if (isNaN(months)) {
+                return error.value;
+            }
+            months = parseInt(months, 10);
+            return serial(new Date(start_date.getFullYear(), start_date.getMonth() + months + 1, 0));
+        };
+    
+        exports.HOUR = function(serial_number) {
+            serial_number = utils.parseDate(serial_number);
+            if (serial_number instanceof Error) {
+                return serial_number;
+            }
+            return serial_number.getHours();
+        };
+    
+        exports.INTERVAL = function(second) {
+            if (typeof second !== 'number' && typeof second !== 'string') {
+                return error.value;
+            } else {
+                second = parseInt(second, 10);
+            }
+    
+            var year  = Math.floor(second/946080000);
+            second    = second%946080000;
+            var month = Math.floor(second/2592000);
+            second    = second%2592000;
+            var day   = Math.floor(second/86400);
+            second    = second%86400;
+    
+            var hour  = Math.floor(second/3600);
+            second    = second%3600;
+            var min   = Math.floor(second/60);
+            second    = second%60;
+            var sec   = second;
+    
+            year  = (year  > 0) ? year  + 'Y' : '';
+            month = (month > 0) ? month + 'M' : '';
+            day   = (day   > 0) ? day   + 'D' : '';
+            hour  = (hour  > 0) ? hour  + 'H' : '';
+            min   = (min   > 0) ? min   + 'M' : '';
+            sec   = (sec   > 0) ? sec   + 'S' : '';
+    
+            return 'P' + year + month + day + 'T' + hour + min + sec;
+        };
+    
+        exports.ISOWEEKNUM = function(date) {
+            date = utils.parseDate(date);
+            if (date instanceof Error) {
+                return date;
+            }
+    
+            date.setHours(0, 0, 0);
+            date.setDate(date.getDate() + 4 - (date.getDay() || 7));
+            var yearStart = new Date(date.getFullYear(), 0, 1);
+            return Math.ceil((((date - yearStart) / 86400000) + 1) / 7);
+        };
+    
+        exports.MINUTE = function(serial_number) {
+            serial_number = utils.parseDate(serial_number);
+            if (serial_number instanceof Error) {
+                return serial_number;
+            }
+            return serial_number.getMinutes();
+        };
+    
+        exports.MONTH = function(serial_number) {
+            serial_number = utils.parseDate(serial_number);
+            if (serial_number instanceof Error) {
+                return serial_number;
+            }
+            return serial_number.getMonth() + 1;
+        };
+    
+        exports.NETWORKDAYS = function(start_date, end_date, holidays) {
+        };
+    
+        exports.NETWORKDAYS.INTL = function(start_date, end_date, weekend, holidays) {
+        };
+    
+        exports.NOW = function() {
+            return new Date();
+        };
+    
+        exports.SECOND = function(serial_number) {
+            serial_number = utils.parseDate(serial_number);
+            if (serial_number instanceof Error) {
+                return serial_number;
+            }
+            return serial_number.getSeconds();
+        };
+    
+        exports.TIME = function(hour, minute, second) {
+            hour = utils.parseNumber(hour);
+            minute = utils.parseNumber(minute);
+            second = utils.parseNumber(second);
+            if (utils.anyIsError(hour, minute, second)) {
+                return error.value;
+            }
+            if (hour < 0 || minute < 0 || second < 0) {
+                return error.num;
+            }
+            return (3600 * hour + 60 * minute + second) / 86400;
+        };
+    
+        exports.TIMEVALUE = function(time_text) {
+            time_text = utils.parseDate(time_text);
+            if (time_text instanceof Error) {
+                return time_text;
+            }
+            return (3600 * time_text.getHours() + 60 * time_text.getMinutes() + time_text.getSeconds()) / 86400;
+        };
+    
+        exports.TODAY = function() {
+            return new Date();
+        };
+    
+        exports.WEEKDAY = function(serial_number, return_type) {
+            serial_number = utils.parseDate(serial_number);
+            if (serial_number instanceof Error) {
+                return serial_number;
+            }
+            if (return_type === undefined) {
+                return_type = 1;
+            }
+            var day = serial_number.getDay();
+            return WEEK_TYPES[return_type][day];
+        };
+    
+        exports.WEEKNUM = function(serial_number, return_type) {
+        };
+    
+        exports.WORKDAY = function(start_date, days, holidays) {
+        };
+    
+        exports.WORKDAY.INTL = function(start_date, days, weekend, holidays) {
+        };
+    
+        exports.YEAR = function(serial_number) {
+            serial_number = utils.parseDate(serial_number);
+            if (serial_number instanceof Error) {
+                return serial_number;
+            }
+            return serial_number.getFullYear();
+        };
+    
+        function isLeapYear(year) {
+            return new Date(year, 1, 29).getMonth() === 1;
+        }
+    
+        exports.YEARFRAC = function(start_date, end_date, basis) {
+        };
+    
+        function serial(date) {
+            var addOn = (date > -2203891200000)?2:1;
+            return (date - d1900) / 86400000 + addOn;
+        }
+    
+        return exports;
+    })();
+    
+    jexcel.methods.database = (function() {
+        var exports = {};
+    
+        function compact(array) {
+            if (!array) {
+                return array;
+            }
+            var result = [];
+            for (var i = 0; i < array.length; ++i) {
+                if (!array[i]) {
+                    continue;
+                }
+                result.push(array[i]);
+            }
+            return result;
+        }
+    
+        exports.FINDFIELD = function(database, title) {
+            var index = null;
+            for (var i = 0; i < database.length; i++) {
+                if (database[i][0] === title) {
+                    index = i;
+                    break;
+                }
+            }
+    
+            // Return error if the input field title is incorrect
+            if (index == null) {
+                return error.value;
+            }
+            return index;
+        };
+    
+        function findResultIndex(database, criterias) {
+            var matches = {};
+            for (var i = 1; i < database[0].length; ++i) {
+                matches[i] = true;
+            }
+            var maxCriteriaLength = criterias[0].length;
+            for (i = 1; i < criterias.length; ++i) {
+                if (criterias[i].length > maxCriteriaLength) {
+                    maxCriteriaLength = criterias[i].length;
+                }
+            }
+    
+            for (var k = 1; k < database.length; ++k) {
+                for (var l = 1; l < database[k].length; ++l) {
+                    var currentCriteriaResult = false;
+                    var hasMatchingCriteria = false;
+                    for (var j = 0; j < criterias.length; ++j) {
+                        var criteria = criterias[j];
+                        if (criteria.length < maxCriteriaLength) {
+                            continue;
+                        }
+    
+                        var criteriaField = criteria[0];
+                        if (database[k][0] !== criteriaField) {
+                            continue;
+                        }
+                        hasMatchingCriteria = true;
+                        for (var p = 1; p < criteria.length; ++p) {
+                            currentCriteriaResult = currentCriteriaResult
+                                    || eval(database[k][l] + criteria[p]); // jshint
+                                                                            // ignore:line
+                        }
+                    }
+                    if (hasMatchingCriteria) {
+                        matches[l] = matches[l] && currentCriteriaResult;
+                    }
+                }
+            }
+    
+            var result = [];
+            for (var n = 0; n < database[0].length; ++n) {
+                if (matches[n]) {
+                    result.push(n - 1);
+                }
+            }
+            return result;
+        }
+    
+        // Database functions
+        exports.DAVERAGE = function(database, field, criteria) {
+            // Return error if field is not a number and not a string
+            if (isNaN(field) && (typeof field !== "string")) {
+                return error.value;
+            }
+            var resultIndexes = findResultIndex(database, criteria);
+            var targetFields = [];
+            if (typeof field === "string") {
+                var index = exports.FINDFIELD(database, field);
+                targetFields = utils.rest(database[index]);
+            } else {
+                targetFields = utils.rest(database[field]);
+            }
+            var sum = 0;
+            for (var i = 0; i < resultIndexes.length; i++) {
+                sum += targetFields[resultIndexes[i]];
+            }
+            return resultIndexes.length === 0 ? error.div0 : sum / resultIndexes.length;
+        };
+    
+        exports.DCOUNT = function(database, field, criteria) {
+        };
+    
+        exports.DCOUNTA = function(database, field, criteria) {
+        };
+    
+        exports.DGET = function(database, field, criteria) {
+            // Return error if field is not a number and not a string
+            if (isNaN(field) && (typeof field !== "string")) {
+                return error.value;
+            }
+            var resultIndexes = findResultIndex(database, criteria);
+            var targetFields = [];
+            if (typeof field === "string") {
+                var index = exports.FINDFIELD(database, field);
+                targetFields = utils.rest(database[index]);
+            } else {
+                targetFields = utils.rest(database[field]);
+            }
+            // Return error if no record meets the criteria
+            if (resultIndexes.length === 0) {
+                return error.value;
+            }
+            // Returns the #NUM! error value because more than one record meets the
+            // criteria
+            if (resultIndexes.length > 1) {
+                return error.num;
+            }
+    
+            return targetFields[resultIndexes[0]];
+        };
+    
+        exports.DMAX = function(database, field, criteria) {
+            // Return error if field is not a number and not a string
+            if (isNaN(field) && (typeof field !== "string")) {
+                return error.value;
+            }
+            var resultIndexes = findResultIndex(database, criteria);
+            var targetFields = [];
+            if (typeof field === "string") {
+                var index = exports.FINDFIELD(database, field);
+                targetFields = utils.rest(database[index]);
+            } else {
+                targetFields = utils.rest(database[field]);
+            }
+            var maxValue = targetFields[resultIndexes[0]];
+            for (var i = 1; i < resultIndexes.length; i++) {
+                if (maxValue < targetFields[resultIndexes[i]]) {
+                    maxValue = targetFields[resultIndexes[i]];
+                }
+            }
+            return maxValue;
+        };
+    
+        exports.DMIN = function(database, field, criteria) {
+            // Return error if field is not a number and not a string
+            if (isNaN(field) && (typeof field !== "string")) {
+                return error.value;
+            }
+            var resultIndexes = findResultIndex(database, criteria);
+            var targetFields = [];
+            if (typeof field === "string") {
+                var index = exports.FINDFIELD(database, field);
+                targetFields = utils.rest(database[index]);
+            } else {
+                targetFields = utils.rest(database[field]);
+            }
+            var minValue = targetFields[resultIndexes[0]];
+            for (var i = 1; i < resultIndexes.length; i++) {
+                if (minValue > targetFields[resultIndexes[i]]) {
+                    minValue = targetFields[resultIndexes[i]];
+                }
+            }
+            return minValue;
+        };
+    
+        exports.DPRODUCT = function(database, field, criteria) {
+            // Return error if field is not a number and not a string
+            if (isNaN(field) && (typeof field !== "string")) {
+                return error.value;
+            }
+            var resultIndexes = findResultIndex(database, criteria);
+            var targetFields = [];
+            if (typeof field === "string") {
+                var index = exports.FINDFIELD(database, field);
+                targetFields = utils.rest(database[index]);
+            } else {
+                targetFields = utils.rest(database[field]);
+            }
+            var targetValues = [];
+            for (var i = 0; i < resultIndexes.length; i++) {
+                targetValues[i] = targetFields[resultIndexes[i]];
+            }
+            targetValues = compact(targetValues);
+            var result = 1;
+            for (i = 0; i < targetValues.length; i++) {
+                result *= targetValues[i];
+            }
+            return result;
+        };
+    
+        exports.DSTDEV = function(database, field, criteria) {
+        };
+    
+        exports.DSTDEVP = function(database, field, criteria) {
+        };
+    
+        exports.DSUM = function(database, field, criteria) {
+        };
+    
+        exports.DVAR = function(database, field, criteria) {
+        };
+    
+        exports.DVARP = function(database, field, criteria) {
+        };
+    
+        exports.MATCH = function(lookupValue, lookupArray, matchType) {
+            if (!lookupValue && !lookupArray) {
+                return error.na;
+            }
+            if (arguments.length === 2) {
+                matchType = 1;
+            }
+            if (!(lookupArray instanceof Array)) {
+                return error.na;
+            }
+            if (matchType !== -1 && matchType !== 0 && matchType !== 1) {
+                return error.na;
+            }
+    
+            var index;
+            var indexValue;
+    
+            for (var idx = 0; idx < lookupArray.length; idx++) {
+                if (matchType === 1) {
+                    if (lookupArray[idx] === lookupValue) {
+                        return idx + 1;
+                    } else if (lookupArray[idx] < lookupValue) {
+                        if (!indexValue) {
+                            index = idx + 1;
+                            indexValue = lookupArray[idx];
+                        } else if (lookupArray[idx] > indexValue) {
+                            index = idx + 1;
+                            indexValue = lookupArray[idx];
+                        }
+                    }
+                } else if (matchType === 0) {
+                    if (typeof lookupValue === 'string') {
+                        lookupValue = lookupValue.replace(/\?/g, '.');
+                        if (lookupArray[idx].toLowerCase().match(lookupValue.toLowerCase())) {
+                            return idx + 1;
+                        }
+                    } else {
+                        if (lookupArray[idx] === lookupValue) {
+                            return idx + 1;
+                        }
+                    }
+                } else if (matchType === -1) {
+                    if (lookupArray[idx] === lookupValue) {
+                        return idx + 1;
+                    } else if (lookupArray[idx] > lookupValue) {
+                        if (!indexValue) {
+                            index = idx + 1;
+                            indexValue = lookupArray[idx];
+                        } else if (lookupArray[idx] < indexValue) {
+                            index = idx + 1;
+                            indexValue = lookupArray[idx];
+                        }
+                    }
+                }
+            }
+    
+            return index ? index : error.na;
+        };
+    
+        return exports;
+    })();
+    
+    jexcel.methods.engineering = (function() {
+        var exports = {};
+    
+        function isValidBinaryNumber(number) {
+            return (/^[01]{1,10}$/).test(number);
+        }
+    
+        exports.BESSELI = function(x, n) {
+        };
+    
+        exports.BESSELJ = function(x, n) {
+        };
+    
+        exports.BESSELK = function(x, n) {
+        };
+    
+        exports.BESSELY = function(x, n) {
+        };
+    
+        exports.BIN2DEC = function(number) {
+            // Return error if number is not binary or contains more than 10
+            // characters (10 digits)
+            if (!isValidBinaryNumber(number)) {
+                return error.num;
+            }
+    
+            // Convert binary number to decimal
+            var result = parseInt(number, 2);
+    
+            // Handle negative numbers
+            var stringified = number.toString();
+            if (stringified.length === 10 && stringified.substring(0, 1) === '1') {
+                return parseInt(stringified.substring(1), 2) - 512;
+            } else {
+                return result;
+            }
+        };
+    
+        exports.BIN2HEX = function(number, places) {
+            // Return error if number is not binary or contains more than 10
+            // characters (10 digits)
+            if (!isValidBinaryNumber(number)) {
+                return error.num;
+            }
+    
+            // Ignore places and return a 10-character hexadecimal number if number
+            // is negative
+            var stringified = number.toString();
+            if (stringified.length === 10 && stringified.substring(0, 1) === '1') {
+                return (1099511627264 + parseInt(stringified.substring(1), 2)).toString(16);
+            }
+    
+            // Convert binary number to hexadecimal
+            var result = parseInt(number, 2).toString(16);
+    
+            // Return hexadecimal number using the minimum number of characters
+            // necessary if places is undefined
+            if (places === undefined) {
+                return result;
+            } else {
+                // Return error if places is nonnumeric
+                if (isNaN(places)) {
+                  return error.value;
+                }
+    
+                // Return error if places is negative
+                if (places < 0) {
+                  return error.num;
+                }
+    
+                // Truncate places in case it is not an integer
+                places = Math.floor(places);
+    
+                // Pad return value with leading 0s (zeros) if necessary (using
+                // Underscore.string)
+                return (places >= result.length) ? REPT('0', places - result.length) + result : error.num;
+            }
+        };
+    
+        exports.BIN2OCT = function(number, places) {
+            // Return error if number is not binary or contains more than 10
+            // characters (10 digits)
+            if (!isValidBinaryNumber(number)) {
+                return error.num;
+            }
+    
+            // Ignore places and return a 10-character octal number if number is
+            // negative
+            var stringified = number.toString();
+            if (stringified.length === 10 && stringified.substring(0, 1) === '1') {
+                return (1073741312 + parseInt(stringified.substring(1), 2)).toString(8);
+            }
+    
+            // Convert binary number to octal
+            var result = parseInt(number, 2).toString(8);
+    
+            // Return octal number using the minimum number of characters necessary
+            // if places is undefined
+            if (places === undefined) {
+                return result;
+            } else {
+                // Return error if places is nonnumeric
+                if (isNaN(places)) {
+                  return error.value;
+                }
+    
+                // Return error if places is negative
+                if (places < 0) {
+                  return error.num;
+                }
+    
+                // Truncate places in case it is not an integer
+                places = Math.floor(places);
+    
+                // Pad return value with leading 0s (zeros) if necessary (using
+                // Underscore.string)
+                return (places >= result.length) ? REPT('0', places - result.length) + result : error.num;
+            }
+        };
+    
+        exports.BITAND = function(number1, number2) {
+            // Return error if either number is a non-numeric value
+            number1 = utils.parseNumber(number1);
+            number2 = utils.parseNumber(number2);
+            if (utils.anyIsError(number1, number2)) {
+                return error.value;
+            }
+    
+            // Return error if either number is less than 0
+            if (number1 < 0 || number2 < 0) {
+                return error.num;
+            }
+    
+            // Return error if either number is a non-integer
+            if (Math.floor(number1) !== number1 || Math.floor(number2) !== number2) {
+                return error.num;
+            }
+    
+            // Return error if either number is greater than (2^48)-1
+            if (number1 > 281474976710655 || number2 > 281474976710655) {
+                return error.num;
+            }
+    
+            // Return bitwise AND of two numbers
+            return number1 & number2;
+        };
+    
+        exports.BITLSHIFT = function(number, shift) {
+            number = utils.parseNumber(number);
+            shift = utils.parseNumber(shift);
+            if (utils.anyIsError(number, shift)) {
+                return error.value;
+            }
+    
+            // Return error if number is less than 0
+            if (number < 0) {
+                return error.num;
+            }
+    
+            // Return error if number is a non-integer
+            if (Math.floor(number) !== number) {
+                return error.num;
+            }
+    
+            // Return error if number is greater than (2^48)-1
+            if (number > 281474976710655) {
+                return error.num;
+            }
+    
+            // Return error if the absolute value of shift is greater than 53
+            if (Math.abs(shift) > 53) {
+                return error.num;
+            }
+    
+            // Return number shifted by shift bits to the left or to the right if
+            // shift is negative
+            return (shift >= 0) ? number << shift : number >> -shift;
+        };
+    
+        exports.BITOR = function(number1, number2) {
+            number1 = utils.parseNumber(number1);
+            number2 = utils.parseNumber(number2);
+            if (utils.anyIsError(number1, number2)) {
+                return error.value;
+            }
+    
+            // Return error if either number is less than 0
+            if (number1 < 0 || number2 < 0) {
+                return error.num;
+            }
+    
+            // Return error if either number is a non-integer
+            if (Math.floor(number1) !== number1 || Math.floor(number2) !== number2) {
+                return error.num;
+            }
+    
+            // Return error if either number is greater than (2^48)-1
+            if (number1 > 281474976710655 || number2 > 281474976710655) {
+                return error.num;
+            }
+    
+            // Return bitwise OR of two numbers
+            return number1 | number2;
+        };
+    
+        exports.BITRSHIFT = function(number, shift) {
+            number = utils.parseNumber(number);
+            shift = utils.parseNumber(shift);
+            if (utils.anyIsError(number, shift)) {
+                return error.value;
+            }
+    
+            // Return error if number is less than 0
+            if (number < 0) {
+                return error.num;
+            }
+    
+            // Return error if number is a non-integer
+            if (Math.floor(number) !== number) {
+                return error.num;
+            }
+    
+            // Return error if number is greater than (2^48)-1
+            if (number > 281474976710655) {
+                return error.num;
+            }
+    
+            // Return error if the absolute value of shift is greater than 53
+            if (Math.abs(shift) > 53) {
+                return error.num;
+            }
+    
+            // Return number shifted by shift bits to the right or to the left if
+            // shift is negative
+            return (shift >= 0) ? number >> shift : number << -shift;
+        };
+    
+        exports.BITXOR = function(number1, number2) {
+            number1 = utils.parseNumber(number1);
+            number2 = utils.parseNumber(number2);
+            if (utils.anyIsError(number1, number2)) {
+                return error.value;
+            }
+    
+            // Return error if either number is less than 0
+            if (number1 < 0 || number2 < 0) {
+                return error.num;
+            }
+    
+            // Return error if either number is a non-integer
+            if (Math.floor(number1) !== number1 || Math.floor(number2) !== number2) {
+                return error.num;
+            }
+    
+            // Return error if either number is greater than (2^48)-1
+            if (number1 > 281474976710655 || number2 > 281474976710655) {
+                return error.num;
+            }
+    
+            // Return bitwise XOR of two numbers
+            return number1 ^ number2;
+        };
+    
+        exports.COMPLEX = function(real, imaginary, suffix) {
+            real = utils.parseNumber(real);
+            imaginary = utils.parseNumber(imaginary);
+            if (utils.anyIsError(real, imaginary)) {
+                return real;
+            }
+    
+            // Set suffix
+            suffix = (suffix === undefined) ? 'i' : suffix;
+    
+            // Return error if suffix is neither "i" nor "j"
+            if (suffix !== 'i' && suffix !== 'j') {
+                return error.value;
+            }
+    
+            // Return complex number
+            if (real === 0 && imaginary === 0) {
+                return 0;
+            } else if (real === 0) {
+                return (imaginary === 1) ? suffix : imaginary.toString() + suffix;
+            } else if (imaginary === 0) {
+                return real.toString();
+            } else {
+                var sign = (imaginary > 0) ? '+' : '';
+                return real.toString() + sign + ((imaginary === 1) ? suffix : imaginary.toString() + suffix);
+            }
+        };
+    
+        exports.CONVERT = function(number, from_unit, to_unit) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+    
+            // List of units supported by CONVERT and units defined by the
+            // International System of Units
+            // [Name, Symbol, Alternate symbols, Quantity, ISU, CONVERT, Conversion
+            // ratio]
+            var units = [
+                ["a.u. of action", "?", null, "action", false, false, 1.05457168181818e-34],
+                ["a.u. of charge", "e", null, "electric_charge", false, false, 1.60217653141414e-19],
+                ["a.u. of energy", "Eh", null, "energy", false, false, 4.35974417757576e-18],
+                ["a.u. of length", "a?", null, "length", false, false, 5.29177210818182e-11],
+                ["a.u. of mass", "m?", null, "mass", false, false, 9.10938261616162e-31],
+                ["a.u. of time", "?/Eh", null, "time", false, false, 2.41888432650516e-17],
+                ["admiralty knot", "admkn", null, "speed", false, true, 0.514773333],
+                ["ampere", "A", null, "electric_current", true, false, 1],
+                ["ampere per meter", "A/m", null, "magnetic_field_intensity", true, false, 1],
+                ["ångström", "Å", ["ang"], "length", false, true, 1e-10],
+                ["are", "ar", null, "area", false, true, 100],
+                ["astronomical unit", "ua", null, "length", false, false, 1.49597870691667e-11],
+                ["bar", "bar", null, "pressure", false, false, 100000],
+                ["barn", "b", null, "area", false, false, 1e-28],
+                ["becquerel", "Bq", null, "radioactivity", true, false, 1],
+                ["bit", "bit", ["b"], "information", false, true, 1],
+                ["btu", "BTU", ["btu"], "energy", false, true, 1055.05585262],
+                ["byte", "byte", null, "information", false, true, 8],
+                ["candela", "cd", null, "luminous_intensity", true, false, 1],
+                ["candela per square metre", "cd/m?", null, "luminance", true, false, 1],
+                ["coulomb", "C", null, "electric_charge", true, false, 1],
+                ["cubic ångström", "ang3", ["ang^3"], "volume", false, true, 1e-30],
+                ["cubic foot", "ft3", ["ft^3"], "volume", false, true, 0.028316846592],
+                ["cubic inch", "in3", ["in^3"], "volume", false, true, 0.000016387064],
+                ["cubic light-year", "ly3", ["ly^3"], "volume", false, true, 8.46786664623715e-47],
+                ["cubic metre", "m?", null, "volume", true, true, 1],
+                ["cubic mile", "mi3", ["mi^3"], "volume", false, true, 4168181825.44058],
+                ["cubic nautical mile", "Nmi3", ["Nmi^3"], "volume", false, true, 6352182208],
+                ["cubic Pica", "Pica3", ["Picapt3", "Pica^3", "Picapt^3"], "volume", false, true, 7.58660370370369e-8],
+                ["cubic yard", "yd3", ["yd^3"], "volume", false, true, 0.764554857984],
+                ["cup", "cup", null, "volume", false, true, 0.0002365882365],
+                ["dalton", "Da", ["u"], "mass", false, false, 1.66053886282828e-27],
+                ["day", "d", ["day"], "time", false, true, 86400],
+                ["degree", "°", null, "angle", false, false, 0.0174532925199433],
+                ["degrees Rankine", "Rank", null, "temperature", false, true, 0.555555555555556],
+                ["dyne", "dyn", ["dy"], "force", false, true, 0.00001],
+                ["electronvolt", "eV", ["ev"], "energy", false, true, 1.60217656514141],
+                ["ell", "ell", null, "length", false, true, 1.143],
+                ["erg", "erg", ["e"], "energy", false, true, 1e-7],
+                ["farad", "F", null, "electric_capacitance", true, false, 1],
+                ["fluid ounce", "oz", null, "volume", false, true, 0.0000295735295625],
+                ["foot", "ft", null, "length", false, true, 0.3048],
+                ["foot-pound", "flb", null, "energy", false, true, 1.3558179483314],
+                ["gal", "Gal", null, "acceleration", false, false, 0.01],
+                ["gallon", "gal", null, "volume", false, true, 0.003785411784],
+                ["gauss", "G", ["ga"], "magnetic_flux_density", false, true, 1],
+                ["grain", "grain", null, "mass", false, true, 0.0000647989],
+                ["gram", "g", null, "mass", false, true, 0.001],
+                ["gray", "Gy", null, "absorbed_dose", true, false, 1],
+                ["gross registered ton", "GRT", ["regton"], "volume", false, true, 2.8316846592],
+                ["hectare", "ha", null, "area", false, true, 10000],
+                ["henry", "H", null, "inductance", true, false, 1],
+                ["hertz", "Hz", null, "frequency", true, false, 1],
+                ["horsepower", "HP", ["h"], "power", false, true, 745.69987158227],
+                ["horsepower-hour", "HPh", ["hh", "hph"], "energy", false, true, 2684519.538],
+                ["hour", "h", ["hr"], "time", false, true, 3600],
+                ["imperial gallon (U.K.)", "uk_gal", null, "volume", false, true, 0.00454609],
+                ["imperial hundredweight", "lcwt", ["uk_cwt", "hweight"], "mass", false, true, 50.802345],
+                ["imperial quart (U.K)", "uk_qt", null, "volume", false, true, 0.0011365225],
+                ["imperial ton", "brton", ["uk_ton", "LTON"], "mass", false, true, 1016.046909],
+                ["inch", "in", null, "length", false, true, 0.0254],
+                ["international acre", "uk_acre", null, "area", false, true, 4046.8564224],
+                ["IT calorie", "cal", null, "energy", false, true, 4.1868],
+                ["joule", "J", null, "energy", true, true, 1],
+                ["katal", "kat", null, "catalytic_activity", true, false, 1],
+                ["kelvin", "K", ["kel"], "temperature", true, true, 1],
+                ["kilogram", "kg", null, "mass", true, true, 1],
+                ["knot", "kn", null, "speed", false, true, 0.514444444444444],
+                ["light-year", "ly", null, "length", false, true, 9460730472580800],
+                ["litre", "L", ["l", "lt"], "volume", false, true, 0.001],
+                ["lumen", "lm", null, "luminous_flux", true, false, 1],
+                ["lux", "lx", null, "illuminance", true, false, 1],
+                ["maxwell", "Mx", null, "magnetic_flux", false, false, 1e-18],
+                ["measurement ton", "MTON", null, "volume", false, true, 1.13267386368],
+                ["meter per hour", "m/h", ["m/hr"], "speed", false, true, 0.00027777777777778],
+                ["meter per second", "m/s", ["m/sec"], "speed", true, true, 1],
+                ["meter per second squared", "m?s??", null, "acceleration", true, false, 1],
+                ["parsec", "pc", ["parsec"], "length", false, true, 30856775814671900],
+                ["meter squared per second", "m?/s", null, "kinematic_viscosity", true, false, 1],
+                ["metre", "m", null, "length", true, true, 1],
+                ["miles per hour", "mph", null, "speed", false, true, 0.44704],
+                ["millimetre of mercury", "mmHg", null, "pressure", false, false, 133.322],
+                ["minute", "?", null, "angle", false, false, 0.000290888208665722],
+                ["minute", "min", ["mn"], "time", false, true, 60],
+                ["modern teaspoon", "tspm", null, "volume", false, true, 0.000005],
+                ["mole", "mol", null, "amount_of_substance", true, false, 1],
+                ["morgen", "Morgen", null, "area", false, true, 2500],
+                ["n.u. of action", "?", null, "action", false, false, 1.05457168181818e-34],
+                ["n.u. of mass", "m?", null, "mass", false, false, 9.10938261616162e-31],
+                ["n.u. of speed", "c?", null, "speed", false, false, 299792458],
+                ["n.u. of time", "?/(me?c??)", null, "time", false, false, 1.28808866778687e-21],
+                ["nautical mile", "M", ["Nmi"], "length", false, true, 1852],
+                ["newton", "N", null, "force", true, true, 1],
+                ["Å“rsted", "Oe ", null, "magnetic_field_intensity", false, false, 79.5774715459477],
+                ["ohm", "Ω", null, "electric_resistance", true, false, 1],
+                ["ounce mass", "ozm", null, "mass", false, true, 0.028349523125],
+                ["pascal", "Pa", null, "pressure", true, false, 1],
+                ["pascal second", "Pa?s", null, "dynamic_viscosity", true, false, 1],
+                ["pferdestärke", "PS", null, "power", false, true, 735.49875],
+                ["phot", "ph", null, "illuminance", false, false, 0.0001],
+                ["pica (1/6 inch)", "pica", null, "length", false, true, 0.00035277777777778],
+                ["pica (1/72 inch)", "Pica", ["Picapt"], "length", false, true, 0.00423333333333333],
+                ["poise", "P", null, "dynamic_viscosity", false, false, 0.1],
+                ["pond", "pond", null, "force", false, true, 0.00980665],
+                ["pound force", "lbf", null, "force", false, true, 4.4482216152605],
+                ["pound mass", "lbm", null, "mass", false, true, 0.45359237],
+                ["quart", "qt", null, "volume", false, true, 0.000946352946],
+                ["radian", "rad", null, "angle", true, false, 1],
+                ["second", "?", null, "angle", false, false, 0.00000484813681109536],
+                ["second", "s", ["sec"], "time", true, true, 1],
+                ["short hundredweight", "cwt", ["shweight"], "mass", false, true, 45.359237],
+                ["siemens", "S", null, "electrical_conductance", true, false, 1],
+                ["sievert", "Sv", null, "equivalent_dose", true, false, 1],
+                ["slug", "sg", null, "mass", false, true, 14.59390294],
+                ["square ångström", "ang2", ["ang^2"], "area", false, true, 1e-20],
+                ["square foot", "ft2", ["ft^2"], "area", false, true, 0.09290304],
+                ["square inch", "in2", ["in^2"], "area", false, true, 0.00064516],
+                ["square light-year", "ly2", ["ly^2"], "area", false, true, 8.95054210748189e+31],
+                ["square meter", "m?", null, "area", true, true, 1],
+                ["square mile", "mi2", ["mi^2"], "area", false, true, 2589988.110336],
+                ["square nautical mile", "Nmi2", ["Nmi^2"], "area", false, true, 3429904],
+                ["square Pica", "Pica2", ["Picapt2", "Pica^2", "Picapt^2"], "area", false, true, 0.00001792111111111],
+                ["square yard", "yd2", ["yd^2"], "area", false, true, 0.83612736],
+                ["statute mile", "mi", null, "length", false, true, 1609.344],
+                ["steradian", "sr", null, "solid_angle", true, false, 1],
+                ["stilb", "sb", null, "luminance", false, false, 0.0001],
+                ["stokes", "St", null, "kinematic_viscosity", false, false, 0.0001],
+                ["stone", "stone", null, "mass", false, true, 6.35029318],
+                ["tablespoon", "tbs", null, "volume", false, true, 0.0000147868],
+                ["teaspoon", "tsp", null, "volume", false, true, 0.00000492892],
+                ["tesla", "T", null, "magnetic_flux_density", true, true, 1],
+                ["thermodynamic calorie", "c", null, "energy", false, true, 4.184],
+                ["ton", "ton", null, "mass", false, true, 907.18474],
+                ["tonne", "t", null, "mass", false, false, 1000],
+                ["U.K. pint", "uk_pt", null, "volume", false, true, 0.00056826125],
+                ["U.S. bushel", "bushel", null, "volume", false, true, 0.03523907],
+                ["U.S. oil barrel", "barrel", null, "volume", false, true, 0.158987295],
+                ["U.S. pint", "pt", ["us_pt"], "volume", false, true, 0.000473176473],
+                ["U.S. survey mile", "survey_mi", null, "length", false, true, 1609.347219],
+                ["U.S. survey/statute acre", "us_acre", null, "area", false, true, 4046.87261],
+                ["volt", "V", null, "voltage", true, false, 1],
+                ["watt", "W", null, "power", true, true, 1],
+                ["watt-hour", "Wh", ["wh"], "energy", false, true, 3600],
+                ["weber", "Wb", null, "magnetic_flux", true, false, 1],
+                ["yard", "yd", null, "length", false, true, 0.9144],
+                ["year", "yr", null, "time", false, true, 31557600]
+            ];
+    
+            // Binary prefixes
+            // [Name, Prefix power of 2 value, Previx value, Abbreviation, Derived
+            // from]
+            var binary_prefixes = {
+                Yi: ["yobi", 80, 1208925819614629174706176, "Yi", "yotta"],
+                Zi: ["zebi", 70, 1180591620717411303424, "Zi", "zetta"],
+                Ei: ["exbi", 60, 1152921504606846976, "Ei", "exa"],
+                Pi: ["pebi", 50, 1125899906842624, "Pi", "peta"],
+                Ti: ["tebi", 40, 1099511627776, "Ti", "tera"],
+                Gi: ["gibi", 30, 1073741824, "Gi", "giga"],
+                Mi: ["mebi", 20, 1048576, "Mi", "mega"],
+                ki: ["kibi", 10, 1024, "ki", "kilo"]
+            };
+    
+            // Unit prefixes
+            // [Name, Multiplier, Abbreviation]
+            var unit_prefixes = {
+                Y: ["yotta", 1e+24, "Y"],
+                Z: ["zetta", 1e+21, "Z"],
+                E: ["exa", 1e+18, "E"],
+                P: ["peta", 1e+15, "P"],
+                T: ["tera", 1e+12, "T"],
+                G: ["giga", 1e+09, "G"],
+                M: ["mega", 1e+06, "M"],
+                k: ["kilo", 1e+03, "k"],
+                h: ["hecto", 1e+02, "h"],
+                e: ["dekao", 1e+01, "e"],
+                d: ["deci", 1e-01, "d"],
+                c: ["centi", 1e-02, "c"],
+                m: ["milli", 1e-03, "m"],
+                u: ["micro", 1e-06, "u"],
+                n: ["nano", 1e-09, "n"],
+                p: ["pico", 1e-12, "p"],
+                f: ["femto", 1e-15, "f"],
+                a: ["atto", 1e-18, "a"],
+                z: ["zepto", 1e-21, "z"],
+                y: ["yocto", 1e-24, "y"]
+            };
+    
+            // Initialize units and multipliers
+            var from = null;
+            var to = null;
+            var base_from_unit = from_unit;
+            var base_to_unit = to_unit;
+            var from_multiplier = 1;
+            var to_multiplier = 1;
+            var alt;
+    
+            // Lookup from and to units
+            for (var i = 0; i < units.length; i++) {
+                alt = (units[i][2] === null) ? [] : units[i][2];
+                if (units[i][1] === base_from_unit || alt.indexOf(base_from_unit) >= 0) {
+                  from = units[i];
+                }
+                if (units[i][1] === base_to_unit || alt.indexOf(base_to_unit) >= 0) {
+                  to = units[i];
+                }
+            }
+    
+            // Lookup from prefix
+            if (from === null) {
+                var from_binary_prefix = binary_prefixes[from_unit.substring(0, 2)];
+                var from_unit_prefix = unit_prefixes[from_unit.substring(0, 1)];
+    
+                // Handle dekao unit prefix (only unit prefix with two characters)
+                if (from_unit.substring(0, 2) === 'da') {
+                  from_unit_prefix = ["dekao", 1e+01, "da"];
+                }
+    
+                // Handle binary prefixes first (so that 'Yi' is processed before
+                // 'Y')
+                if (from_binary_prefix) {
+                  from_multiplier = from_binary_prefix[2];
+                  base_from_unit = from_unit.substring(2);
+                } else if (from_unit_prefix) {
+                  from_multiplier = from_unit_prefix[1];
+                  base_from_unit = from_unit.substring(from_unit_prefix[2].length);
+                }
+    
+                // Lookup from unit
+                for (var j = 0; j < units.length; j++) {
+                  alt = (units[j][2] === null) ? [] : units[j][2];
+                  if (units[j][1] === base_from_unit || alt.indexOf(base_from_unit) >= 0) {
+                      from = units[j];
+                  }
+                }
+            }
+    
+            // Lookup to prefix
+            if (to === null) {
+                var to_binary_prefix = binary_prefixes[to_unit.substring(0, 2)];
+                var to_unit_prefix = unit_prefixes[to_unit.substring(0, 1)];
+    
+                // Handle dekao unit prefix (only unit prefix with two characters)
+                if (to_unit.substring(0, 2) === 'da') {
+                  to_unit_prefix = ["dekao", 1e+01, "da"];
+                }
+    
+                // Handle binary prefixes first (so that 'Yi' is processed before
+                // 'Y')
+                if (to_binary_prefix) {
+                  to_multiplier = to_binary_prefix[2];
+                  base_to_unit = to_unit.substring(2);
+                } else if (to_unit_prefix) {
+                  to_multiplier = to_unit_prefix[1];
+                  base_to_unit = to_unit.substring(to_unit_prefix[2].length);
+                }
+    
+                // Lookup to unit
+                for (var k = 0; k < units.length; k++) {
+                  alt = (units[k][2] === null) ? [] : units[k][2];
+                  if (units[k][1] === base_to_unit || alt.indexOf(base_to_unit) >= 0) {
+                      to = units[k];
+                  }
+                }
+            }
+    
+            // Return error if a unit does not exist
+            if (from === null || to === null) {
+                return error.na;
+            }
+    
+            // Return error if units represent different quantities
+            if (from[3] !== to[3]) {
+                return error.na;
+            }
+    
+            // Return converted number
+            return number * from[6] * from_multiplier / (to[6] * to_multiplier);
+        };
+    
+        exports.DEC2BIN = function(number, places) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+    
+            // Return error if number is not decimal, is lower than -512, or is
+            // greater than 511
+            if (!/^-?[0-9]{1,3}$/.test(number) || number < -512 || number > 511) {
+                return error.num;
+            }
+    
+            // Ignore places and return a 10-character binary number if number is
+            // negative
+            if (number < 0) {
+                return '1' + REPT('0', 9 - (512 + number).toString(2).length) + (512 + number).toString(2);
+            }
+    
+            // Convert decimal number to binary
+            var result = parseInt(number, 10).toString(2);
+    
+            // Return binary number using the minimum number of characters necessary
+            // if places is undefined
+            if (typeof places === 'undefined') {
+                return result;
+            } else {
+                // Return error if places is nonnumeric
+                if (isNaN(places)) {
+                  return error.value;
+                }
+    
+                // Return error if places is negative
+                if (places < 0) {
+                  return error.num;
+                }
+    
+                // Truncate places in case it is not an integer
+                places = Math.floor(places);
+    
+                // Pad return value with leading 0s (zeros) if necessary (using
+                // Underscore.string)
+                return (places >= result.length) ? REPT('0', places - result.length) + result : error.num;
+            }
+        };
+    
+        exports.DEC2HEX = function(number, places) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+    
+            // Return error if number is not decimal, is lower than -549755813888,
+            // or is greater than 549755813887
+            if (!/^-?[0-9]{1,12}$/.test(number) || number < -549755813888 || number > 549755813887) {
+                return error.num;
+            }
+    
+            // Ignore places and return a 10-character hexadecimal number if number
+            // is negative
+            if (number < 0) {
+                return (1099511627776 + number).toString(16);
+            }
+    
+            // Convert decimal number to hexadecimal
+            var result = parseInt(number, 10).toString(16);
+    
+            // Return hexadecimal number using the minimum number of characters
+            // necessary if places is undefined
+            if (typeof places === 'undefined') {
+                return result;
+            } else {
+                // Return error if places is nonnumeric
+                if (isNaN(places)) {
+                  return error.value;
+                }
+    
+                // Return error if places is negative
+                if (places < 0) {
+                  return error.num;
+                }
+    
+                // Truncate places in case it is not an integer
+                places = Math.floor(places);
+    
+                // Pad return value with leading 0s (zeros) if necessary (using
+                // Underscore.string)
+                return (places >= result.length) ? REPT('0', places - result.length) + result : error.num;
+            }
+        };
+    
+        exports.DEC2OCT = function(number, places) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+    
+            // Return error if number is not decimal, is lower than -549755813888,
+            // or is greater than 549755813887
+            if (!/^-?[0-9]{1,9}$/.test(number) || number < -536870912 || number > 536870911) {
+                return error.num;
+            }
+    
+            // Ignore places and return a 10-character octal number if number is
+            // negative
+            if (number < 0) {
+                return (1073741824 + number).toString(8);
+            }
+    
+            // Convert decimal number to octal
+            var result = parseInt(number, 10).toString(8);
+    
+            // Return octal number using the minimum number of characters necessary
+            // if places is undefined
+            if (typeof places === 'undefined') {
+                return result;
+            } else {
+                // Return error if places is nonnumeric
+                if (isNaN(places)) {
+                  return error.value;
+                }
+    
+                // Return error if places is negative
+                if (places < 0) {
+                  return error.num;
+                }
+    
+                // Truncate places in case it is not an integer
+                places = Math.floor(places);
+    
+                // Pad return value with leading 0s (zeros) if necessary (using
+                // Underscore.string)
+                return (places >= result.length) ? REPT('0', places - result.length) + result : error.num;
+            }
+        };
+    
+        exports.DELTA = function(number1, number2) {
+            // Set number2 to zero if undefined
+            number2 = (number2 === undefined) ? 0 : number2;
+            number1 = utils.parseNumber(number1);
+            number2 = utils.parseNumber(number2);
+            if (utils.anyIsError(number1, number2)) {
+                return error.value;
+            }
+    
+            // Return delta
+            return (number1 === number2) ? 1 : 0;
+        };
+    
+        exports.ERF = function(lower_bound, upper_bound) {
+        };
+    
+        exports.ERF.PRECISE = function() {
+        };
+    
+        exports.ERFC = function(x) {
+        };
+    
+        exports.ERFC.PRECISE = function() {
+        };
+    
+        exports.GESTEP = function(number, step) {
+            step = step || 0;
+            number = utils.parseNumber(number);
+            if (utils.anyIsError(step, number)) {
+                return number;
+            }
+    
+            // Return delta
+            return (number >= step) ? 1 : 0;
+        };
+    
+        exports.HEX2BIN = function(number, places) {
+            // Return error if number is not hexadecimal or contains more than ten
+            // characters (10 digits)
+            if (!/^[0-9A-Fa-f]{1,10}$/.test(number)) {
+                return error.num;
+            }
+    
+            // Check if number is negative
+            var negative = (number.length === 10 && number.substring(0, 1).toLowerCase() === 'f') ? true : false;
+    
+            // Convert hexadecimal number to decimal
+            var decimal = (negative) ? parseInt(number, 16) - 1099511627776 : parseInt(number, 16);
+    
+            // Return error if number is lower than -512 or greater than 511
+            if (decimal < -512 || decimal > 511) {
+                return error.num;
+            }
+    
+            // Ignore places and return a 10-character binary number if number is
+            // negative
+            if (negative) {
+                return '1' + REPT('0', 9 - (512 + decimal).toString(2).length) + (512 + decimal).toString(2);
+            }
+    
+            // Convert decimal number to binary
+            var result = decimal.toString(2);
+    
+            // Return binary number using the minimum number of characters necessary
+            // if places is undefined
+            if (places === undefined) {
+                return result;
+            } else {
+                // Return error if places is nonnumeric
+                if (isNaN(places)) {
+                  return error.value;
+                }
+    
+                // Return error if places is negative
+                if (places < 0) {
+                  return error.num;
+                }
+    
+                // Truncate places in case it is not an integer
+                places = Math.floor(places);
+    
+                // Pad return value with leading 0s (zeros) if necessary (using
+                // Underscore.string)
+                return (places >= result.length) ? REPT('0', places - result.length) + result : error.num;
+            }
+        };
+    
+        exports.HEX2DEC = function(number) {
+            // Return error if number is not hexadecimal or contains more than ten
+            // characters (10 digits)
+            if (!/^[0-9A-Fa-f]{1,10}$/.test(number)) {
+                return error.num;
+            }
+    
+            // Convert hexadecimal number to decimal
+            var decimal = parseInt(number, 16);
+    
+            // Return decimal number
+            return (decimal >= 549755813888) ? decimal - 1099511627776 : decimal;
+        };
+    
+        exports.HEX2OCT = function(number, places) {
+            // Return error if number is not hexadecimal or contains more than ten
+            // characters (10 digits)
+            if (!/^[0-9A-Fa-f]{1,10}$/.test(number)) {
+                return error.num;
+            }
+    
+            // Convert hexadecimal number to decimal
+            var decimal = parseInt(number, 16);
+    
+            // Return error if number is positive and greater than 0x1fffffff
+            // (536870911)
+            if (decimal > 536870911 && decimal < 1098974756864) {
+                return error.num;
+            }
+    
+            // Ignore places and return a 10-character octal number if number is
+            // negative
+            if (decimal >= 1098974756864) {
+                return (decimal - 1098437885952).toString(8);
+            }
+    
+            // Convert decimal number to octal
+            var result = decimal.toString(8);
+    
+            // Return octal number using the minimum number of characters necessary
+            // if places is undefined
+            if (places === undefined) {
+                return result;
+            } else {
+                // Return error if places is nonnumeric
+                if (isNaN(places)) {
+                  return error.value;
+                }
+    
+                // Return error if places is negative
+                if (places < 0) {
+                  return error.num;
+                }
+    
+                // Truncate places in case it is not an integer
+                places = Math.floor(places);
+    
+                // Pad return value with leading 0s (zeros) if necessary (using
+                // Underscore.string)
+                return (places >= result.length) ? REPT('0', places - result.length) + result : error.num;
+            }
+        };
+    
+        exports.IMABS = function(inumber) {
+            // Lookup real and imaginary coefficients using exports.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            // Return error if either coefficient is not a number
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Return absolute value of complex number
+            return Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2));
+        };
+    
+        exports.IMAGINARY = function(inumber) {
+            if (inumber === undefined || inumber === true || inumber === false) {
+                return error.value;
+            }
+    
+            // Return 0 if inumber is equal to 0
+            if (inumber === 0 || inumber === '0') {
+                return 0;
+            }
+    
+            // Handle special cases
+            if (['i', 'j'].indexOf(inumber) >= 0) {
+                return 1;
+            }
+    
+            // Normalize imaginary coefficient
+            inumber = inumber.replace('+i', '+1i').replace('-i', '-1i').replace('+j', '+1j').replace('-j', '-1j');
+    
+            // Lookup sign
+            var plus = inumber.indexOf('+');
+            var minus = inumber.indexOf('-');
+            if (plus === 0) {
+                plus = inumber.indexOf('+', 1);
+            }
+    
+            if (minus === 0) {
+                minus = inumber.indexOf('-', 1);
+            }
+    
+            // Lookup imaginary unit
+            var last = inumber.substring(inumber.length - 1, inumber.length);
+            var unit = (last === 'i' || last === 'j');
+    
+            if (plus >= 0 || minus >= 0) {
+                // Return error if imaginary unit is neither i nor j
+                if (!unit) {
+                  return error.num;
+                }
+    
+                // Return imaginary coefficient of complex number
+                if (plus >= 0) {
+                  return (isNaN(inumber.substring(0, plus)) || isNaN(inumber.substring(plus + 1, inumber.length - 1))) ?
+                      error.num :
+                      Number(inumber.substring(plus + 1, inumber.length - 1));
+                } else {
+                  return (isNaN(inumber.substring(0, minus)) || isNaN(inumber.substring(minus + 1, inumber.length - 1))) ?
+                      error.num :
+                      -Number(inumber.substring(minus + 1, inumber.length - 1));
+                }
+            } else {
+                if (unit) {
+                  return (isNaN(inumber.substring(0, inumber.length - 1))) ? error.num : inumber.substring(0, inumber.length - 1);
+                } else {
+                  return (isNaN(inumber)) ? error.num : 0;
+                }
+            }
+        };
+    
+        exports.IMARGUMENT = function(inumber) {
+            // Lookup real and imaginary coefficients using exports.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            // Return error if either coefficient is not a number
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Return error if inumber is equal to zero
+            if (x === 0 && y === 0) {
+                return error.div0;
+            }
+    
+            // Return PI/2 if x is equal to zero and y is positive
+            if (x === 0 && y > 0) {
+                return Math.PI / 2;
+            }
+    
+            // Return -PI/2 if x is equal to zero and y is negative
+            if (x === 0 && y < 0) {
+                return -Math.PI / 2;
+            }
+    
+            // Return zero if x is negative and y is equal to zero
+            if (y === 0 && x > 0) {
+                return 0;
+            }
+    
+            // Return zero if x is negative and y is equal to zero
+            if (y === 0 && x < 0) {
+                return -Math.PI;
+            }
+    
+            // Return argument of complex number
+            if (x > 0) {
+                return Math.atan(y / x);
+            } else if (x < 0 && y >= 0) {
+                return Math.atan(y / x) + Math.PI;
+            } else {
+                return Math.atan(y / x) - Math.PI;
+            }
+        };
+    
+        exports.IMCONJUGATE = function(inumber) {
+            // Lookup real and imaginary coefficients using exports.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Return conjugate of complex number
+            return (y !== 0) ? exports.COMPLEX(x, -y, unit) : inumber;
+        };
+    
+        exports.IMCOS = function(inumber) {
+            // Lookup real and imaginary coefficients using exports.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Return cosine of complex number
+            return exports.COMPLEX(Math.cos(x) * (Math.exp(y) + Math.exp(-y)) / 2, -Math.sin(x) * (Math.exp(y) - Math.exp(-y)) / 2, unit);
+        };
+    
+        exports.IMCOSH = function(inumber) {
+            // Lookup real and imaginary coefficients using exports.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Return hyperbolic cosine of complex number
+            return exports.COMPLEX(Math.cos(y) * (Math.exp(x) + Math.exp(-x)) / 2, Math.sin(y) * (Math.exp(x) - Math.exp(-x)) / 2, unit);
+        };
+    
+        exports.IMCOT = function(inumber) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Return cotangent of complex number
+            return exports.IMDIV(exports.IMCOS(inumber), exports.IMSIN(inumber));
+        };
+    
+        exports.IMDIV = function(inumber1, inumber2) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var a = exports.IMREAL(inumber1);
+            var b = exports.IMAGINARY(inumber1);
+            var c = exports.IMREAL(inumber2);
+            var d = exports.IMAGINARY(inumber2);
+    
+            if (utils.anyIsError(a, b, c, d)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit1 = inumber1.substring(inumber1.length - 1);
+            var unit2 = inumber2.substring(inumber2.length - 1);
+            var unit = 'i';
+            if (unit1 === 'j') {
+                unit = 'j';
+            } else if (unit2 === 'j') {
+                unit = 'j';
+            }
+    
+            // Return error if inumber2 is null
+            if (c === 0 && d === 0) {
+                return error.num;
+            }
+    
+            // Return exponential of complex number
+            var den = c * c + d * d;
+            return exports.COMPLEX((a * c + b * d) / den, (b * c - a * d) / den, unit);
+        };
+    
+        exports.IMEXP = function(inumber) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Return exponential of complex number
+            var e = Math.exp(x);
+            return exports.COMPLEX(e * Math.cos(y), e * Math.sin(y), unit);
+        };
+    
+        exports.IMLN = function(inumber) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Return exponential of complex number
+            return exports.COMPLEX(Math.log(Math.sqrt(x * x + y * y)), Math.atan(y / x), unit);
+        };
+    
+        exports.IMLOG10 = function(inumber) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Return exponential of complex number
+            return exports.COMPLEX(Math.log(Math.sqrt(x * x + y * y)) / Math.log(10), Math.atan(y / x) / Math.log(10), unit);
+        };
+    
+        exports.IMLOG2 = function(inumber) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Return exponential of complex number
+            return exports.COMPLEX(Math.log(Math.sqrt(x * x + y * y)) / Math.log(2), Math.atan(y / x) / Math.log(2), unit);
+        };
+    
+        exports.IMPOWER = function(inumber, number) {
+            number = utils.parseNumber(number);
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+            if (utils.anyIsError(number, x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Calculate power of modulus
+            var p = Math.pow(exports.IMABS(inumber), number);
+    
+            // Calculate argument
+            var t = exports.IMARGUMENT(inumber);
+    
+            // Return exponential of complex number
+            return exports.COMPLEX(p * Math.cos(number * t), p * Math.sin(number * t), unit);
+        };
+    
+        exports.IMPRODUCT = function() {
+            // Initialize result
+            var result = arguments[0];
+    
+            // Loop on all numbers
+            for (var i = 1; i < arguments.length; i++) {
+                // Lookup coefficients of two complex numbers
+                var a = exports.IMREAL(result);
+                var b = exports.IMAGINARY(result);
+                var c = exports.IMREAL(arguments[i]);
+                var d = exports.IMAGINARY(arguments[i]);
+    
+                if (utils.anyIsError(a, b, c, d)) {
+                  return error.value;
+                }
+    
+                // Complute product of two complex numbers
+                result = exports.COMPLEX(a * c - b * d, a * d + b * c);
+            }
+    
+            // Return product of complex numbers
+            return result;
+        };
+    
+        exports.IMREAL = function(inumber) {
+            if (inumber === undefined || inumber === true || inumber === false) {
+                return error.value;
+            }
+    
+            // Return 0 if inumber is equal to 0
+            if (inumber === 0 || inumber === '0') {
+                return 0;
+            }
+    
+            // Handle special cases
+            if (['i', '+i', '1i', '+1i', '-i', '-1i', 'j', '+j', '1j', '+1j', '-j', '-1j'].indexOf(inumber) >= 0) {
+                return 0;
+            }
+    
+            // Lookup sign
+            var plus = inumber.indexOf('+');
+            var minus = inumber.indexOf('-');
+            if (plus === 0) {
+                plus = inumber.indexOf('+', 1);
+            }
+            if (minus === 0) {
+                minus = inumber.indexOf('-', 1);
+            }
+    
+            // Lookup imaginary unit
+            var last = inumber.substring(inumber.length - 1, inumber.length);
+            var unit = (last === 'i' || last === 'j');
+    
+            if (plus >= 0 || minus >= 0) {
+                // Return error if imaginary unit is neither i nor j
+                if (!unit) {
+                  return error.num;
+                }
+    
+                // Return real coefficient of complex number
+                if (plus >= 0) {
+                  return (isNaN(inumber.substring(0, plus)) || isNaN(inumber.substring(plus + 1, inumber.length - 1))) ?
+                      error.num :
+                      Number(inumber.substring(0, plus));
+                } else {
+                  return (isNaN(inumber.substring(0, minus)) || isNaN(inumber.substring(minus + 1, inumber.length - 1))) ?
+                      error.num :
+                      Number(inumber.substring(0, minus));
+                }
+            } else {
+                if (unit) {
+                  return (isNaN(inumber.substring(0, inumber.length - 1))) ? error.num : 0;
+                } else {
+                  return (isNaN(inumber)) ? error.num : inumber;
+                }
+            }
+        };
+    
+        exports.IMSEC = function(inumber) {
+            // Return error if inumber is a logical value
+            if (inumber === true || inumber === false) {
+                return error.value;
+            }
+    
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Return secant of complex number
+            return exports.IMDIV('1', exports.IMCOS(inumber));
+        };
+    
+        exports.IMSECH = function(inumber) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Return hyperbolic secant of complex number
+            return exports.IMDIV('1', exports.IMCOSH(inumber));
+        };
+    
+        exports.IMSIN = function(inumber) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Return sine of complex number
+            return exports.COMPLEX(Math.sin(x) * (Math.exp(y) + Math.exp(-y)) / 2, Math.cos(x) * (Math.exp(y) - Math.exp(-y)) / 2, unit);
+        };
+    
+        exports.IMSINH = function(inumber) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Return hyperbolic sine of complex number
+            return exports.COMPLEX(Math.cos(y) * (Math.exp(x) - Math.exp(-x)) / 2, Math.sin(y) * (Math.exp(x) + Math.exp(-x)) / 2, unit);
+        };
+    
+        exports.IMSQRT = function(inumber) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit = inumber.substring(inumber.length - 1);
+            unit = (unit === 'i' || unit === 'j') ? unit : 'i';
+    
+            // Calculate power of modulus
+            var s = Math.sqrt(exports.IMABS(inumber));
+    
+            // Calculate argument
+            var t = exports.IMARGUMENT(inumber);
+    
+            // Return exponential of complex number
+            return exports.COMPLEX(s * Math.cos(t / 2), s * Math.sin(t / 2), unit);
+        };
+    
+        exports.IMCSC = function (inumber) {
+            // Return error if inumber is a logical value
+            if (inumber === true || inumber === false) {
+                return error.value;
+            }
+    
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            // Return error if either coefficient is not a number
+            if (utils.anyIsError(x, y)) {
+                return error.num;
+            }
+    
+            // Return cosecant of complex number
+            return exports.IMDIV('1', exports.IMSIN(inumber));
+        };
+    
+        exports.IMCSCH = function (inumber) {
+            // Return error if inumber is a logical value
+            if (inumber === true || inumber === false) {
+                return error.value;
+            }
+    
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            // Return error if either coefficient is not a number
+            if (utils.anyIsError(x, y)) {
+                return error.num;
+            }
+    
+            // Return hyperbolic cosecant of complex number
+            return exports.IMDIV('1', exports.IMSINH(inumber));
+        };
+    
+        exports.IMSUB = function(inumber1, inumber2) {
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var a = this.IMREAL(inumber1);
+            var b = this.IMAGINARY(inumber1);
+            var c = this.IMREAL(inumber2);
+            var d = this.IMAGINARY(inumber2);
+    
+            if (utils.anyIsError(a, b, c, d)) {
+                return error.value;
+            }
+    
+            // Lookup imaginary unit
+            var unit1 = inumber1.substring(inumber1.length - 1);
+            var unit2 = inumber2.substring(inumber2.length - 1);
+            var unit = 'i';
+            if (unit1 === 'j') {
+                unit = 'j';
+            } else if (unit2 === 'j') {
+                unit = 'j';
+            }
+    
+            // Return _ of two complex numbers
+            return this.COMPLEX(a - c, b - d, unit);
+        };
+    
+        exports.IMSUM = function() {
+            var args = utils.flatten(arguments);
+    
+            // Initialize result
+            var result = args[0];
+    
+            // Loop on all numbers
+            for (var i = 1; i < args.length; i++) {
+                // Lookup coefficients of two complex numbers
+                var a = this.IMREAL(result);
+                var b = this.IMAGINARY(result);
+                var c = this.IMREAL(args[i]);
+                var d = this.IMAGINARY(args[i]);
+    
+                if (utils.anyIsError(a, b, c, d)) {
+                  return error.value;
+                }
+    
+                // Complute product of two complex numbers
+                result = this.COMPLEX(a + c, b + d);
+            }
+    
+            // Return sum of complex numbers
+            return result;
+        };
+    
+        exports.IMTAN = function(inumber) {
+            // Return error if inumber is a logical value
+            if (inumber === true || inumber === false) {
+                return error.value;
+            }
+    
+            // Lookup real and imaginary coefficients using Formula.js
+            // [http://formulajs.org]
+            var x = exports.IMREAL(inumber);
+            var y = exports.IMAGINARY(inumber);
+    
+            if (utils.anyIsError(x, y)) {
+                return error.value;
+            }
+    
+            // Return tangent of complex number
+            return this.IMDIV(this.IMSIN(inumber), this.IMCOS(inumber));
+        };
+    
+        exports.OCT2BIN = function(number, places) {
+            // Return error if number is not hexadecimal or contains more than ten
+            // characters (10 digits)
+            if (!/^[0-7]{1,10}$/.test(number)) {
+                return error.num;
+            }
+    
+            // Check if number is negative
+            var negative = (number.length === 10 && number.substring(0, 1) === '7') ? true : false;
+    
+            // Convert octal number to decimal
+            var decimal = (negative) ? parseInt(number, 8) - 1073741824 : parseInt(number, 8);
+    
+            // Return error if number is lower than -512 or greater than 511
+            if (decimal < -512 || decimal > 511) {
+                return error.num;
+            }
+    
+            // Ignore places and return a 10-character binary number if number is
+            // negative
+            if (negative) {
+                return '1' + REPT('0', 9 - (512 + decimal).toString(2).length) + (512 + decimal).toString(2);
+            }
+    
+            // Convert decimal number to binary
+            var result = decimal.toString(2);
+    
+            // Return binary number using the minimum number of characters necessary
+            // if places is undefined
+            if (typeof places === 'undefined') {
+                return result;
+            } else {
+                // Return error if places is nonnumeric
+                if (isNaN(places)) {
+                  return error.value;
+                }
+    
+                // Return error if places is negative
+                if (places < 0) {
+                  return error.num;
+                }
+    
+                // Truncate places in case it is not an integer
+                places = Math.floor(places);
+    
+                // Pad return value with leading 0s (zeros) if necessary (using
+                // Underscore.string)
+                return (places >= result.length) ? REPT('0', places - result.length) + result : error.num;
+            }
+        };
+    
+        exports.OCT2DEC = function(number) {
+            // Return error if number is not octal or contains more than ten
+            // characters (10 digits)
+            if (!/^[0-7]{1,10}$/.test(number)) {
+                return error.num;
+            }
+    
+            // Convert octal number to decimal
+            var decimal = parseInt(number, 8);
+    
+            // Return decimal number
+            return (decimal >= 536870912) ? decimal - 1073741824 : decimal;
+        };
+    
+        exports.OCT2HEX = function(number, places) {
+            // Return error if number is not octal or contains more than ten
+            // characters (10 digits)
+            if (!/^[0-7]{1,10}$/.test(number)) {
+                return error.num;
+            }
+    
+            // Convert octal number to decimal
+            var decimal = parseInt(number, 8);
+    
+            // Ignore places and return a 10-character octal number if number is
+            // negative
+            if (decimal >= 536870912) {
+                return 'ff' + (decimal + 3221225472).toString(16);
+            }
+    
+            // Convert decimal number to hexadecimal
+            var result = decimal.toString(16);
+    
+            // Return hexadecimal number using the minimum number of characters
+            // necessary if places is undefined
+            if (places === undefined) {
+                return result;
+            } else {
+                // Return error if places is nonnumeric
+                if (isNaN(places)) {
+                  return error.value;
+                }
+    
+                // Return error if places is negative
+                if (places < 0) {
+                  return error.num;
+                }
+    
+                // Truncate places in case it is not an integer
+                places = Math.floor(places);
+    
+                // Pad return value with leading 0s (zeros) if necessary (using
+                // Underscore.string)
+                return (places >= result.length) ? REPT('0', places - result.length) + result : error.num;
+            }
+        };
+    
+        return exports;
+    })();
+    
+    jexcel.methods.financial = (function() {
+        var exports = {};
+    
+        function validDate(d) {
+            return d && d.getTime && !isNaN(d.getTime());
+        }
+    
+        function ensureDate(d) {
+            return (d instanceof Date)?d:new Date(d);
+        }
+    
+        exports.ACCRINT = function(issue, first, settlement, rate, par, frequency, basis) {
+            // Return error if either date is invalid
+            issue        = ensureDate(issue);
+            first        = ensureDate(first);
+            settlement = ensureDate(settlement);
+            if (!validDate(issue) || !validDate(first) || !validDate(settlement)) {
+                return '#VALUE!';
+            }
+    
+            // Return error if either rate or par are lower than or equal to zero
+            if (rate <= 0 || par <= 0) {
+                return '#NUM!';
+            }
+    
+            // Return error if frequency is neither 1, 2, or 4
+            if ([1, 2, 4].indexOf(frequency) === -1) {
+                return '#NUM!';
+            }
+    
+            // Return error if basis is neither 0, 1, 2, 3, or 4
+            if ([0, 1, 2, 3, 4].indexOf(basis) === -1) {
+                return '#NUM!';
+            }
+    
+            // Return error if settlement is before or equal to issue
+            if (settlement <= issue) {
+                return '#NUM!';
+            }
+    
+            // Set default values
+            par   = par   || 0;
+            basis = basis || 0;
+    
+            // Compute accrued interest
+            return par * rate * YEARFRAC(issue, settlement, basis);
+        };
+    
+        exports.ACCRINTM = null;
+    
+        exports.AMORDEGRC = null;
+    
+        exports.AMORLINC = null;
+    
+        exports.COUPDAYBS = null;
+    
+        exports.COUPDAYS = null;
+    
+        exports.COUPDAYSNC = null;
+    
+        exports.COUPNCD = null;
+    
+        exports.COUPNUM = null;
+    
+        exports.COUPPCD = null;
+    
+        exports.CUMIPMT = function(rate, periods, value, start, end, type) {
+            // Credits: algorithm inspired by Apache OpenOffice
+            // Credits: Hannes Stiebitzhofer for the translations of function and
+                // variable names
+            // Requires exports.FV() and exports.PMT() from exports.js
+                // [http://stoic.com/exports/]
+    
+            rate = utils.parseNumber(rate);
+            periods = utils.parseNumber(periods);
+            value = utils.parseNumber(value);
+            if (utils.anyIsError(rate, periods, value)) {
+                return error.value;
+            }
+    
+            // Return error if either rate, periods, or value are lower than or
+                // equal to zero
+            if (rate <= 0 || periods <= 0 || value <= 0) {
+                return error.num;
+            }
+    
+            // Return error if start < 1, end < 1, or start > end
+            if (start < 1 || end < 1 || start > end) {
+                return error.num;
+            }
+    
+            // Return error if type is neither 0 nor 1
+            if (type !== 0 && type !== 1) {
+                return error.num;
+            }
+    
+            // Compute cumulative interest
+            var payment = exports.PMT(rate, periods, value, 0, type);
+            var interest = 0;
+    
+            if (start === 1) {
+                if (type === 0) {
+                    interest = -value;
+                    start++;
+                }
+            }
+    
+            for (var i = start; i <= end; i++) {
+                if (type === 1) {
+                    interest += exports.FV(rate, i - 2, payment, value, 1) - payment;
+                } else {
+                    interest += exports.FV(rate, i - 1, payment, value, 0);
+                }
+            }
+            interest *= rate;
+    
+            // Return cumulative interest
+            return interest;
+        };
+    
+        exports.CUMPRINC = function(rate, periods, value, start, end, type) {
+            // Credits: algorithm inspired by Apache OpenOffice
+            // Credits: Hannes Stiebitzhofer for the translations of function and
+                // variable names
+    
+            rate = utils.parseNumber(rate);
+            periods = utils.parseNumber(periods);
+            value = utils.parseNumber(value);
+            if (utils.anyIsError(rate, periods, value)) {
+                return error.value;
+            }
+    
+            // Return error if either rate, periods, or value are lower than or
+                // equal to zero
+            if (rate <= 0 || periods <= 0 || value <= 0) {
+                return error.num;
+            }
+    
+            // Return error if start < 1, end < 1, or start > end
+            if (start < 1 || end < 1 || start > end) {
+                return error.num;
+            }
+    
+            // Return error if type is neither 0 nor 1
+            if (type !== 0 && type !== 1) {
+                return error.num;
+            }
+    
+            // Compute cumulative principal
+            var payment = exports.PMT(rate, periods, value, 0, type);
+            var principal = 0;
+            if (start === 1) {
+                if (type === 0) {
+                    principal = payment + value * rate;
+                } else {
+                    principal = payment;
+                }
+                start++;
+            }
+            for (var i = start; i <= end; i++) {
+                if (type > 0) {
+                    principal += payment - (exports.FV(rate, i - 2, payment, value, 1) - payment) * rate;
+                } else {
+                    principal += payment - exports.FV(rate, i - 1, payment, value, 0) * rate;
+                }
+            }
+    
+            // Return cumulative principal
+            return principal;
+        };
+    
+        exports.DB = function(cost, salvage, life, period, month) {
+            // Initialize month
+            month = (month === undefined) ? 12 : month;
+    
+            cost = utils.parseNumber(cost);
+            salvage = utils.parseNumber(salvage);
+            life = utils.parseNumber(life);
+            period = utils.parseNumber(period);
+            month = utils.parseNumber(month);
+            if (utils.anyIsError(cost, salvage, life, period, month)) {
+                return error.value;
+            }
+    
+            // Return error if any of the parameters is negative
+            if (cost < 0 || salvage < 0 || life < 0 || period < 0) {
+                return error.num;
+            }
+    
+            // Return error if month is not an integer between 1 and 12
+            if ([1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12].indexOf(month) === -1) {
+                return error.num;
+            }
+    
+            // Return error if period is greater than life
+            if (period > life) {
+                return error.num;
+            }
+    
+            // Return 0 (zero) if salvage is greater than or equal to cost
+            if (salvage >= cost) {
+                return 0;
+            }
+    
+            // Rate is rounded to three decimals places
+            var rate = (1 - Math.pow(salvage / cost, 1 / life)).toFixed(3);
+    
+            // Compute initial depreciation
+            var initial = cost * rate * month / 12;
+    
+            // Compute total depreciation
+            var total = initial;
+            var current = 0;
+            var ceiling = (period === life) ? life - 1 : period;
+            for (var i = 2; i <= ceiling; i++) {
+                current = (cost - total) * rate;
+                total += current;
+            }
+    
+            // Depreciation for the first and last periods are special cases
+            if (period === 1) {
+                // First period
+                return initial;
+            } else if (period === life) {
+                // Last period
+                return (cost - total) * rate;
+            } else {
+                return current;
+            }
+        };
+    
+        exports.DDB = function(cost, salvage, life, period, factor) {
+            // Initialize factor
+            factor = (factor === undefined) ? 2 : factor;
+    
+            cost = utils.parseNumber(cost);
+            salvage = utils.parseNumber(salvage);
+            life = utils.parseNumber(life);
+            period = utils.parseNumber(period);
+            factor = utils.parseNumber(factor);
+            if (utils.anyIsError(cost, salvage, life, period, factor)) {
+                return error.value;
+            }
+    
+            // Return error if any of the parameters is negative or if factor is
+                // null
+            if (cost < 0 || salvage < 0 || life < 0 || period < 0 || factor <= 0) {
+                return error.num;
+            }
+    
+            // Return error if period is greater than life
+            if (period > life) {
+                return error.num;
+            }
+    
+            // Return 0 (zero) if salvage is greater than or equal to cost
+            if (salvage >= cost) {
+                return 0;
+            }
+    
+            // Compute depreciation
+            var total = 0;
+            var current = 0;
+            for (var i = 1; i <= period; i++) {
+                current = Math.min((cost - total) * (factor / life), (cost - salvage - total));
+                total += current;
+            }
+    
+            // Return depreciation
+            return current;
+        };
+    
+        exports.DISC = null;
+    
+        exports.DOLLARDE = function(dollar, fraction) {
+            // Credits: algorithm inspired by Apache OpenOffice
+    
+            dollar = utils.parseNumber(dollar);
+            fraction = utils.parseNumber(fraction);
+            if (utils.anyIsError(dollar, fraction)) {
+                return error.value;
+            }
+    
+            // Return error if fraction is negative
+            if (fraction < 0) {
+                return error.num;
+            }
+    
+            // Return error if fraction is greater than or equal to 0 and less than
+                // 1
+            if (fraction >= 0 && fraction < 1) {
+                return error.div0;
+            }
+    
+            // Truncate fraction if it is not an integer
+            fraction = parseInt(fraction, 10);
+    
+            // Compute integer part
+            var result = parseInt(dollar, 10);
+    
+            // Add decimal part
+            result += (dollar % 1) * Math.pow(10, Math.ceil(Math.log(fraction) / Math.LN10)) / fraction;
+    
+            // Round result
+            var power = Math.pow(10, Math.ceil(Math.log(fraction) / Math.LN2) + 1);
+            result = Math.round(result * power) / power;
+    
+            // Return converted dollar price
+            return result;
+        };
+    
+        exports.DOLLARFR = function(dollar, fraction) {
+            // Credits: algorithm inspired by Apache OpenOffice
+    
+            dollar = utils.parseNumber(dollar);
+            fraction = utils.parseNumber(fraction);
+            if (utils.anyIsError(dollar, fraction)) {
+                return error.value;
+            }
+    
+            // Return error if fraction is negative
+            if (fraction < 0) {
+                return error.num;
+            }
+    
+            // Return error if fraction is greater than or equal to 0 and less than
+                // 1
+            if (fraction >= 0 && fraction < 1) {
+                return error.div0;
+            }
+    
+            // Truncate fraction if it is not an integer
+            fraction = parseInt(fraction, 10);
+    
+            // Compute integer part
+            var result = parseInt(dollar, 10);
+    
+            // Add decimal part
+            result += (dollar % 1) * Math.pow(10, -Math.ceil(Math.log(fraction) / Math.LN10)) * fraction;
+    
+            // Return converted dollar price
+            return result;
+        };
+    
+        exports.DURATION = null;
+    
+        exports.EFFECT = function(rate, periods) {
+            rate = utils.parseNumber(rate);
+            periods = utils.parseNumber(periods);
+            if (utils.anyIsError(rate, periods)) {
+                return error.value;
+            }
+    
+            // Return error if rate <=0 or periods < 1
+            if (rate <= 0 || periods < 1) {
+                return error.num;
+            }
+    
+            // Truncate periods if it is not an integer
+            periods = parseInt(periods, 10);
+    
+            // Return effective annual interest rate
+            return Math.pow(1 + rate / periods, periods) - 1;
+        };
+    
+        exports.FV = function(rate, periods, payment, value, type) {
+            // Credits: algorithm inspired by Apache OpenOffice
+    
+            value = value || 0;
+            type = type || 0;
+    
+            rate = utils.parseNumber(rate);
+            periods = utils.parseNumber(periods);
+            payment = utils.parseNumber(payment);
+            value = utils.parseNumber(value);
+            type = utils.parseNumber(type);
+            if (utils.anyIsError(rate, periods, payment, value, type)) {
+                return error.value;
+            }
+    
+            // Return future value
+            var result;
+            if (rate === 0) {
+                result = value + payment * periods;
+            } else {
+                var term = Math.pow(1 + rate, periods);
+                if (type === 1) {
+                    result = value * term + payment * (1 + rate) * (term - 1) / rate;
+                } else {
+                    result = value * term + payment * (term - 1) / rate;
+                }
+            }
+            return -result;
+        };
+    
+        exports.FVSCHEDULE = function(principal, schedule) {
+            principal = utils.parseNumber(principal);
+            schedule = utils.parseNumberArray(utils.flatten(schedule));
+            if (utils.anyIsError(principal, schedule)) {
+                return error.value;
+            }
+    
+            var n = schedule.length;
+            var future = principal;
+    
+            // Apply all interests in schedule
+            for (var i = 0; i < n; i++) {
+                // Apply scheduled interest
+                future *= 1 + schedule[i];
+            }
+    
+            // Return future value
+            return future;
+        };
+    
+        exports.INTRATE = null;
+    
+        exports.IPMT = function(rate, period, periods, present, future, type) {
+            // Credits: algorithm inspired by Apache OpenOffice
+    
+            future = future || 0;
+            type = type || 0;
+    
+            rate = utils.parseNumber(rate);
+            period = utils.parseNumber(period);
+            periods = utils.parseNumber(periods);
+            present = utils.parseNumber(present);
+            future = utils.parseNumber(future);
+            type = utils.parseNumber(type);
+            if (utils.anyIsError(rate, period, periods, present, future, type)) {
+                return error.value;
+            }
+    
+            // Compute payment
+            var payment = exports.PMT(rate, periods, present, future, type);
+    
+            // Compute interest
+            var interest;
+            if (period === 1) {
+                if (type === 1) {
+                    interest = 0;
+                } else {
+                    interest = -present;
+                }
+            } else {
+                if (type === 1) {
+                    interest = exports.FV(rate, period - 2, payment, present, 1) - payment;
+                } else {
+                    interest = exports.FV(rate, period - 1, payment, present, 0);
+                }
+            }
+    
+            // Return interest
+            return interest * rate;
+        };
+    
+        exports.IRR = function(values, guess) {
+            // Credits: algorithm inspired by Apache OpenOffice
+    
+            guess = guess || 0;
+    
+            values = utils.parseNumberArray(utils.flatten(values));
+            guess = utils.parseNumber(guess);
+            if (utils.anyIsError(values, guess)) {
+                return error.value;
+            }
+    
+            // Calculates the resulting amount
+            var irrResult = function(values, dates, rate) {
+                var r = rate + 1;
+                var result = values[0];
+                for (var i = 1; i < values.length; i++) {
+                    result += values[i] / Math.pow(r, (dates[i] - dates[0]) / 365);
+                }
+                return result;
+            };
+    
+            // Calculates the first derivation
+            var irrResultDeriv = function(values, dates, rate) {
+                var r = rate + 1;
+                var result = 0;
+                for (var i = 1; i < values.length; i++) {
+                    var frac = (dates[i] - dates[0]) / 365;
+                    result -= frac * values[i] / Math.pow(r, frac + 1);
+                }
+                return result;
+            };
+    
+            // Initialize dates and check that values contains at least one positive
+                // value and one negative value
+            var dates = [];
+            var positive = false;
+            var negative = false;
+            for (var i = 0; i < values.length; i++) {
+                dates[i] = (i === 0) ? 0 : dates[i - 1] + 365;
+                if (values[i] > 0) {
+                    positive = true;
+                }
+                if (values[i] < 0) {
+                    negative = true;
+                }
+            }
+    
+            // Return error if values does not contain at least one positive value
+                // and one negative value
+            if (!positive || !negative) {
+                return error.num;
+            }
+    
+            // Initialize guess and resultRate
+            guess = (guess === undefined) ? 0.1 : guess;
+            var resultRate = guess;
+    
+            // Set maximum epsilon for end of iteration
+            var epsMax = 1e-10;
+    
+            // Implement Newton's method
+            var newRate, epsRate, resultValue;
+            var contLoop = true;
+            do {
+                resultValue = irrResult(values, dates, resultRate);
+                newRate = resultRate - resultValue / irrResultDeriv(values, dates, resultRate);
+                epsRate = Math.abs(newRate - resultRate);
+                resultRate = newRate;
+                contLoop = (epsRate > epsMax) && (Math.abs(resultValue) > epsMax);
+            } while (contLoop);
+    
+            // Return internal rate of return
+            return resultRate;
+        };
+    
+        exports.ISPMT = function(rate, period, periods, value) {
+            rate = utils.parseNumber(rate);
+            period = utils.parseNumber(period);
+            periods = utils.parseNumber(periods);
+            value = utils.parseNumber(value);
+            if (utils.anyIsError(rate, period, periods, value)) {
+                return error.value;
+            }
+    
+            // Return interest
+            return value * rate * (period / periods - 1);
+        };
+    
+        exports.MDURATION = null;
+    
+        exports.MIRR = function(values, finance_rate, reinvest_rate) {
+            values = utils.parseNumberArray(utils.flatten(values));
+            finance_rate = utils.parseNumber(finance_rate);
+            reinvest_rate = utils.parseNumber(reinvest_rate);
+            if (utils.anyIsError(values, finance_rate, reinvest_rate)) {
+                return error.value;
+            }
+    
+            // Initialize number of values
+            var n = values.length;
+    
+            // Lookup payments (negative values) and incomes (positive values)
+            var payments = [];
+            var incomes = [];
+            for (var i = 0; i < n; i++) {
+                if (values[i] < 0) {
+                    payments.push(values[i]);
+                } else {
+                    incomes.push(values[i]);
+                }
+            }
+    
+            // Return modified internal rate of return
+            var num = -exports.NPV(reinvest_rate, incomes) * Math.pow(1 + reinvest_rate, n - 1);
+            var den = exports.NPV(finance_rate, payments) * (1 + finance_rate);
+            return Math.pow(num / den, 1 / (n - 1)) - 1;
+        };
+    
+        exports.NOMINAL = function(rate, periods) {
+            rate = utils.parseNumber(rate);
+            periods = utils.parseNumber(periods);
+            if (utils.anyIsError(rate, periods)) {
+                return error.value;
+            }
+    
+            // Return error if rate <=0 or periods < 1
+            if (rate <= 0 || periods < 1) {
+                return error.num;
+            }
+    
+            // Truncate periods if it is not an integer
+            periods = parseInt(periods, 10);
+    
+            // Return nominal annual interest rate
+            return (Math.pow(rate + 1, 1 / periods) - 1) * periods;
+        };
+    
+        exports.NPER = function(rate, payment, present, future, type) {
+            type = (type === undefined) ? 0 : type;
+            future = (future === undefined) ? 0 : future;
+    
+            rate = utils.parseNumber(rate);
+            payment = utils.parseNumber(payment);
+            present = utils.parseNumber(present);
+            future = utils.parseNumber(future);
+            type = utils.parseNumber(type);
+            if (utils.anyIsError(rate, payment, present, future, type)) {
+                return error.value;
+            }
+    
+            // Return number of periods
+            var num = payment * (1 + rate * type) - future * rate;
+            var den = (present * rate + payment * (1 + rate * type));
+            return Math.log(num / den) / Math.log(1 + rate);
+        };
+    
+        exports.NPV = function() {
+            var args = utils.parseNumberArray(utils.flatten(arguments));
+            if (args instanceof Error) {
+                return args;
+            }
+    
+            // Lookup rate
+            var rate = args[0];
+    
+            // Initialize net present value
+            var value = 0;
+    
+            // Loop on all values
+            for (var j = 1; j < args.length; j++) {
+                value += args[j] / Math.pow(1 + rate, j);
+            }
+    
+            // Return net present value
+            return value;
+        };
+    
+        exports.ODDFPRICE = null;
+    
+        exports.ODDFYIELD = null;
+    
+        exports.ODDLPRICE = null;
+    
+        exports.ODDLYIELD = null;
+    
+        exports.PDURATION = function(rate, present, future) {
+            rate = utils.parseNumber(rate);
+            present = utils.parseNumber(present);
+            future = utils.parseNumber(future);
+            if (utils.anyIsError(rate, present, future)) {
+                return error.value;
+            }
+    
+            // Return error if rate <=0
+            if (rate <= 0) {
+                return error.num;
+            }
+    
+            // Return number of periods
+            return (Math.log(future) - Math.log(present)) / Math.log(1 + rate);
+        };
+    
+        exports.PMT = function(rate, periods, present, future, type) {
+            // Credits: algorithm inspired by Apache OpenOffice
+    
+            future = future || 0;
+            type = type || 0;
+    
+            rate = utils.parseNumber(rate);
+            periods = utils.parseNumber(periods);
+            present = utils.parseNumber(present);
+            future = utils.parseNumber(future);
+            type = utils.parseNumber(type);
+            if (utils.anyIsError(rate, periods, present, future, type)) {
+                return error.value;
+            }
+    
+            // Return payment
+            var result;
+            if (rate === 0) {
+                result = (present + future) / periods;
+            } else {
+                var term = Math.pow(1 + rate, periods);
+                if (type === 1) {
+                    result = (future * rate / (term - 1) + present * rate / (1 - 1 / term)) / (1 + rate);
+                } else {
+                    result = future * rate / (term - 1) + present * rate / (1 - 1 / term);
+                }
+            }
+            return -result;
+        };
+    
+        exports.PPMT = function(rate, period, periods, present, future, type) {
+            future = future || 0;
+            type = type || 0;
+    
+            rate = utils.parseNumber(rate);
+            periods = utils.parseNumber(periods);
+            present = utils.parseNumber(present);
+            future = utils.parseNumber(future);
+            type = utils.parseNumber(type);
+            if (utils.anyIsError(rate, periods, present, future, type)) {
+                return error.value;
+            }
+    
+            return exports.PMT(rate, periods, present, future, type) - exports.IPMT(rate, period, periods, present, future, type);
+        };
+    
+        exports.PRICE = null;
+    
+        exports.PRICEDISC = null;
+    
+        exports.PRICEMAT = null;
+    
+        exports.PV = function(rate, periods, payment, future, type) {
+            future = future || 0;
+            type = type || 0;
+    
+            rate = utils.parseNumber(rate);
+            periods = utils.parseNumber(periods);
+            payment = utils.parseNumber(payment);
+            future = utils.parseNumber(future);
+            type = utils.parseNumber(type);
+            if (utils.anyIsError(rate, periods, payment, future, type)) {
+                return error.value;
+            }
+    
+            // Return present value
+            if (rate === 0) {
+                return -payment * periods - future;
+            } else {
+                return (((1 - Math.pow(1 + rate, periods)) / rate) * payment * (1 + rate * type) - future) / Math.pow(1 + rate, periods);
+            }
+        };
+    
+        exports.RATE = function(periods, payment, present, future, type, guess) {
+            // Credits: rabugento
+    
+            guess = (guess === undefined) ? 0.01 : guess;
+            future = (future === undefined) ? 0 : future;
+            type = (type === undefined) ? 0 : type;
+    
+            periods = utils.parseNumber(periods);
+            payment = utils.parseNumber(payment);
+            present = utils.parseNumber(present);
+            future = utils.parseNumber(future);
+            type = utils.parseNumber(type);
+            guess = utils.parseNumber(guess);
+            if (utils.anyIsError(periods, payment, present, future, type, guess)) {
+                return error.value;
+            }
+    
+            // Set maximum epsilon for end of iteration
+            var epsMax = 1e-6;
+    
+            // Set maximum number of iterations
+            var iterMax = 100;
+            var iter = 0;
+            var close = false;
+            var rate = guess;
+    
+            while (iter < iterMax && !close) {
+                var t1 = Math.pow(rate + 1, periods);
+                var t2 = Math.pow(rate + 1, periods - 1);
+    
+                var f1 = future + t1 * present + payment * (t1 - 1) * (rate * type + 1) / rate;
+                var f2 = periods * t2 * present - payment * (t1 - 1) *(rate * type + 1) / Math.pow(rate,2);
+                var f3 = periods * payment * t2 * (rate * type + 1) / rate + payment * (t1 - 1) * type / rate;
+    
+                var newRate = rate - f1 / (f2 + f3);
+    
+                if (Math.abs(newRate - rate) < epsMax) close = true;
+                iter++
+                rate = newRate;
+            }
+    
+            if (!close) return Number.NaN + rate;
+            return rate;
+        };
+    
+        // TODO
+        exports.RECEIVED = null;
+    
+        exports.RRI = function(periods, present, future) {
+            periods = utils.parseNumber(periods);
+            present = utils.parseNumber(present);
+            future = utils.parseNumber(future);
+            if (utils.anyIsError(periods, present, future)) {
+                return error.value;
+            }
+    
+            // Return error if periods or present is equal to 0 (zero)
+            if (periods === 0 || present === 0) {
+                return error.num;
+            }
+    
+            // Return equivalent interest rate
+            return Math.pow(future / present, 1 / periods) - 1;
+        };
+    
+        exports.SLN = function(cost, salvage, life) {
+            cost = utils.parseNumber(cost);
+            salvage = utils.parseNumber(salvage);
+            life = utils.parseNumber(life);
+            if (utils.anyIsError(cost, salvage, life)) {
+                return error.value;
+            }
+    
+            // Return error if life equal to 0 (zero)
+            if (life === 0) {
+                return error.num;
+            }
+    
+            // Return straight-line depreciation
+            return (cost - salvage) / life;
+        };
+    
+        exports.SYD = function(cost, salvage, life, period) {
+            // Return error if any of the parameters is not a number
+            cost = utils.parseNumber(cost);
+            salvage = utils.parseNumber(salvage);
+            life = utils.parseNumber(life);
+            period = utils.parseNumber(period);
+            if (utils.anyIsError(cost, salvage, life, period)) {
+                return error.value;
+            }
+    
+            // Return error if life equal to 0 (zero)
+            if (life === 0) {
+                return error.num;
+            }
+    
+            // Return error if period is lower than 1 or greater than life
+            if (period < 1 || period > life) {
+                return error.num;
+            }
+    
+            // Truncate period if it is not an integer
+            period = parseInt(period, 10);
+    
+            // Return straight-line depreciation
+            return ((cost - salvage) * (life - period + 1) * 2) / (life * (life + 1));
+        };
+    
+        exports.TBILLEQ = function(settlement, maturity, discount) {
+            settlement = utils.parseDate(settlement);
+            maturity = utils.parseDate(maturity);
+            discount = utils.parseNumber(discount);
+            if (utils.anyIsError(settlement, maturity, discount)) {
+                return error.value;
+            }
+    
+            // Return error if discount is lower than or equal to zero
+            if (discount <= 0) {
+                return error.num;
+            }
+    
+            // Return error if settlement is greater than maturity
+            if (settlement > maturity) {
+                return error.num;
+            }
+    
+            // Return error if maturity is more than one year after settlement
+            if (maturity - settlement > 365 * 24 * 60 * 60 * 1000) {
+                return error.num;
+            }
+    
+            // Return bond-equivalent yield
+            return (365 * discount) / (360 - discount * DAYS360(settlement, maturity, false));
+        };
+    
+        exports.TBILLPRICE = function(settlement, maturity, discount) {
+            settlement = utils.parseDate(settlement);
+            maturity = utils.parseDate(maturity);
+            discount = utils.parseNumber(discount);
+            if (utils.anyIsError(settlement, maturity, discount)) {
+                return error.value;
+            }
+    
+            // Return error if discount is lower than or equal to zero
+            if (discount <= 0) {
+                return error.num;
+            }
+    
+            // Return error if settlement is greater than maturity
+            if (settlement > maturity) {
+                return error.num;
+            }
+    
+            // Return error if maturity is more than one year after settlement
+            if (maturity - settlement > 365 * 24 * 60 * 60 * 1000) {
+                return error.num;
+            }
+    
+            // Return bond-equivalent yield
+            return 100 * (1 - discount * DAYS360(settlement, maturity, false) / 360);
+        };
+    
+        exports.TBILLYIELD = function(settlement, maturity, price) {
+            settlement = utils.parseDate(settlement);
+            maturity = utils.parseDate(maturity);
+            price = utils.parseNumber(price);
+            if (utils.anyIsError(settlement, maturity, price)) {
+                return error.value;
+            }
+    
+            // Return error if price is lower than or equal to zero
+            if (price <= 0) {
+                return error.num;
+            }
+    
+            // Return error if settlement is greater than maturity
+            if (settlement > maturity) {
+                return error.num;
+            }
+    
+            // Return error if maturity is more than one year after settlement
+            if (maturity - settlement > 365 * 24 * 60 * 60 * 1000) {
+                return error.num;
+            }
+    
+            // Return bond-equivalent yield
+            return (100 - price) * 360 / (price * DAYS360(settlement, maturity, false));
+        };
+    
+        exports.VDB = null;
+    
+        exports.XIRR = function(values, dates, guess) {
+            // Credits: algorithm inspired by Apache OpenOffice
+    
+            values = utils.parseNumberArray(utils.flatten(values));
+            dates = utils.parseDateArray(utils.flatten(dates));
+            guess = utils.parseNumber(guess);
+            if (utils.anyIsError(values, dates, guess)) {
+                return error.value;
+            }
+    
+            // Calculates the resulting amount
+            var irrResult = function(values, dates, rate) {
+                var r = rate + 1;
+                var result = values[0];
+                for (var i = 1; i < values.length; i++) {
+                    result += values[i] / Math.pow(r, DAYS(dates[i], dates[0]) / 365);
+                }
+                return result;
+            };
+    
+            // Calculates the first derivation
+            var irrResultDeriv = function(values, dates, rate) {
+                var r = rate + 1;
+                var result = 0;
+                for (var i = 1; i < values.length; i++) {
+                    var frac = DAYS(dates[i], dates[0]) / 365;
+                    result -= frac * values[i] / Math.pow(r, frac + 1);
+                }
+                return result;
+            };
+    
+            // Check that values contains at least one positive value and one
+                // negative value
+            var positive = false;
+            var negative = false;
+            for (var i = 0; i < values.length; i++) {
+                if (values[i] > 0) {
+                    positive = true;
+                }
+                if (values[i] < 0) {
+                    negative = true;
+                }
+            }
+    
+            // Return error if values does not contain at least one positive value
+                // and one negative value
+            if (!positive || !negative) {
+                return error.num;
+            }
+    
+            // Initialize guess and resultRate
+            guess = guess || 0.1;
+            var resultRate = guess;
+    
+            // Set maximum epsilon for end of iteration
+            var epsMax = 1e-10;
+    
+            // Implement Newton's method
+            var newRate, epsRate, resultValue;
+            var contLoop = true;
+            do {
+                resultValue = irrResult(values, dates, resultRate);
+                newRate = resultRate - resultValue / irrResultDeriv(values, dates, resultRate);
+                epsRate = Math.abs(newRate - resultRate);
+                resultRate = newRate;
+                contLoop = (epsRate > epsMax) && (Math.abs(resultValue) > epsMax);
+            } while (contLoop);
+    
+            // Return internal rate of return
+            return resultRate;
+        };
+    
+        exports.XNPV = function(rate, values, dates) {
+            rate = utils.parseNumber(rate);
+            values = utils.parseNumberArray(utils.flatten(values));
+            dates = utils.parseDateArray(utils.flatten(dates));
+            if (utils.anyIsError(rate, values, dates)) {
+                return error.value;
+            }
+    
+            var result = 0;
+            for (var i = 0; i < values.length; i++) {
+                result += values[i] / Math.pow(1 + rate, DAYS(dates[i], dates[0]) / 365);
+            }
+            return result;
+        };
+    
+        exports.YIELD = null;
+    
+        exports.YIELDDISC = null;
+    
+        exports.YIELDMAT = null;
+    
+        return exports;
+    })();
+    
+    jexcel.methods.information = (function() {
+        var exports = {};
+        exports.CELL = null;
+    
+        exports.ERROR = {};
+        exports.ERROR.TYPE = function(error_val) {
+            switch (error_val) {
+                case error.nil: return 1;
+                case error.div0: return 2;
+                case error.value: return 3;
+                case error.ref: return 4;
+                case error.name: return 5;
+                case error.num: return 6;
+                case error.na: return 7;
+                case error.data: return 8;
+            }
+            return error.na;
+        };
+    
+        exports.INFO = null;
+    
+        exports.ISBLANK = function(value) {
+            return value === null;
+        };
+    
+        exports.ISBINARY = function (number) {
+            return (/^[01]{1,10}$/).test(number);
+        };
+    
+        exports.ISERR = function(value) {
+            return ([error.value, error.ref, error.div0, error.num, error.name, error.nil]).indexOf(value) >= 0 ||
+                (typeof value === 'number' && (isNaN(value) || !isFinite(value)));
+        };
+    
+        exports.ISERROR = function(value) {
+            return exports.ISERR(value) || value === error.na;
+        };
+    
+        exports.ISEVEN = function(number) {
+            return (Math.floor(Math.abs(number)) & 1) ? false : true;
+        };
+    
+        // TODO
+        exports.ISFORMULA = null;
+    
+        exports.ISLOGICAL = function(value) {
+            return value === true || value === false;
+        };
+    
+        exports.ISNA = function(value) {
+            return value === error.na;
+        };
+    
+        exports.ISNONTEXT = function(value) {
+            return typeof(value) !== 'string';
+        };
+    
+        exports.ISNUMBER = function(value) {
+            return typeof(value) === 'number' && !isNaN(value) && isFinite(value);
+        };
+    
+        exports.ISODD = function(number) {
+            return (Math.floor(Math.abs(number)) & 1) ? true : false;
+        };
+    
+        exports.ISREF = null;
+    
+        exports.ISTEXT = function(value) {
+            return typeof(value) === 'string';
+        };
+    
+        exports.N = function(value) {
+            if (this.ISNUMBER(value)) {
+                return value;
+            }
+            if (value instanceof Date) {
+                return value.getTime();
+            }
+            if (value === true) {
+                return 1;
+            }
+            if (value === false) {
+                return 0;
+            }
+            if (this.ISERROR(value)) {
+                return value;
+            }
+            return 0;
+        };
+    
+        exports.NA = function() {
+            return error.na;
+        };
+    
+        exports.SHEET = null;
+    
+        exports.SHEETS = null;
+    
+        exports.TYPE = function(value) {
+            if (this.ISNUMBER(value)) {
+                return 1;
+            }
+            if (this.ISTEXT(value)) {
+                return 2;
+            }
+            if (this.ISLOGICAL(value)) {
+                return 4;
+            }
+            if (this.ISERROR(value)) {
+                return 16;
+            }
+            if (Array.isArray(value)) {
+                return 64;
+            }
+        };
+    
+        return exports;
+    })();
+    
+    jexcel.methods.logical = (function() {
+        var exports = {};
+    
+        exports.AND = function() {
+            var args = utils.flatten(arguments);
+            var result = true;
+            for (var i = 0; i < args.length; i++) {
+                if (!args[i]) {
+                    result = false;
+                }
+            }
+            return result;
+        };
+    
+        exports.CHOOSE = function() {
+            if (arguments.length < 2) {
+                return error.na;
+            }
+    
+            var index = arguments[0];
+            if (index < 1 || index > 254) {
+                return error.value;
+            }
+    
+            if (arguments.length < index + 1) {
+                return error.value;
+            }
+    
+            return arguments[index];
+        };
+    
+        exports.FALSE = function() {
+            return false;
+        };
+    
+        exports.IF = function(test, then_value, otherwise_value) {
+            return test ? then_value : otherwise_value;
+        };
+    
+        exports.IFERROR = function(value, valueIfError) {
+            if (ISERROR(value)) {
+                return valueIfError;
+            }
+            return value;
+        };
+    
+        exports.IFNA = function(value, value_if_na) {
+            return value === error.na ? value_if_na : value;
+        };
+    
+        exports.NOT = function(logical) {
+            return !logical;
+        };
+    
+        exports.OR = function() {
+            var args = utils.flatten(arguments);
+            var result = false;
+            for (var i = 0; i < args.length; i++) {
+                if (args[i]) {
+                    result = true;
+                }
+            }
+            return result;
+        };
+    
+        exports.TRUE = function() {
+            return true;
+        };
+    
+        exports.XOR = function() {
+            var args = utils.flatten(arguments);
+            var result = 0;
+            for (var i = 0; i < args.length; i++) {
+                if (args[i]) {
+                    result++;
+                }
+            }
+            return (Math.floor(Math.abs(result)) & 1) ? true : false;
+        };
+    
+        exports.SWITCH = function() {
+            var result;
+            if (arguments.length > 0)  {
+                var targetValue = arguments[0];
+                var argc = arguments.length - 1;
+                var switchCount = Math.floor(argc / 2);
+                var switchSatisfied = false;
+                var defaultClause = argc % 2 === 0 ? null : arguments[arguments.length - 1];
+    
+                if (switchCount) {
+                    for (var index = 0; index < switchCount; index++) {
+                        if (targetValue === arguments[index * 2 + 1]) {
+                          result = arguments[index * 2 + 2];
+                          switchSatisfied = true;
+                          break;
+                        }
+                    }
+                }
+    
+                if (!switchSatisfied && defaultClause) {
+                    result = defaultClause;
+                }
+            }
+    
+            return result;
+        };
+    
+        return exports;
+    })();
+    
+    jexcel.methods.math = (function() {
+        var exports = {};
+    
+        exports.ABS = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.abs(utils.parseNumber(number));
+        };
+    
+        exports.ACOS = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.acos(number);
+        };
+    
+        exports.ACOSH = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.log(number + Math.sqrt(number * number - 1));
+        };
+    
+        exports.ACOT = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.atan(1 / number);
+        };
+    
+        exports.ACOTH = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return 0.5 * Math.log((number + 1) / (number - 1));
+        };
+    
+        exports.AGGREGATE = null
+    
+        exports.ARABIC = function(text) {
+            // Credits: Rafa? Kukawski
+            if (!/^M*(?:D?C{0,3}|C[MD])(?:L?X{0,3}|X[CL])(?:V?I{0,3}|I[XV])$/.test(text)) {
+                return error.value;
+            }
+            var r = 0;
+            text.replace(/[MDLV]|C[MD]?|X[CL]?|I[XV]?/g, function(i) {
+                r += {
+                    M: 1000,
+                    CM: 900,
+                    D: 500,
+                    CD: 400,
+                    C: 100,
+                    XC: 90,
+                    L: 50,
+                    XL: 40,
+                    X: 10,
+                    IX: 9,
+                    V: 5,
+                    IV: 4,
+                    I: 1
+                }[i];
+            });
+            return r;
+        };
+    
+        exports.ASIN = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.asin(number);
+        };
+    
+        exports.ASINH = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.log(number + Math.sqrt(number * number + 1));
+        };
+    
+        exports.ATAN = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.atan(number);
+        };
+    
+        exports.ATAN2 = function(number_x, number_y) {
+            number_x = utils.parseNumber(number_x);
+            number_y = utils.parseNumber(number_y);
+            if (utils.anyIsError(number_x, number_y)) {
+                return error.value;
+            }
+            return Math.atan2(number_x, number_y);
+        };
+    
+        exports.ATANH = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.log((1 + number) / (1 - number)) / 2;
+        };
+    
+        exports.BASE = function(number, radix, min_length) {
+            min_length = min_length || 0;
+    
+            number = utils.parseNumber(number);
+            radix = utils.parseNumber(radix);
+            min_length = utils.parseNumber(min_length);
+            if (utils.anyIsError(number, radix, min_length)) {
+                return error.value;
+            }
+            min_length = (min_length === undefined) ? 0 : min_length;
+            var result = number.toString(radix);
+            return new Array(Math.max(min_length + 1 - result.length, 0)).join('0') + result;
+        };
+    
+        exports.CEILING = function(number, significance, mode) {
+            significance = (significance === undefined) ? 1 : significance;
+            mode = (mode === undefined) ? 0 : mode;
+    
+            number = utils.parseNumber(number);
+            significance = utils.parseNumber(significance);
+            mode = utils.parseNumber(mode);
+            if (utils.anyIsError(number, significance, mode)) {
+                return error.value;
+            }
+            if (significance === 0) {
+                return 0;
+            }
+    
+            significance = Math.abs(significance);
+            if (number >= 0) {
+                return Math.ceil(number / significance) * significance;
+            } else {
+                if (mode === 0) {
+                    return -1 * Math.floor(Math.abs(number) / significance) * significance;
+                } else {
+                    return -1 * Math.ceil(Math.abs(number) / significance) * significance;
+                }
+            }
+        };
+    
+        exports.CEILING.MATH = exports.CEILING;
+    
+        exports.CEILING.PRECISE = exports.CEILING;
+    
+        exports.COMBIN = function(number, number_chosen) {
+            number = utils.parseNumber(number);
+            number_chosen = utils.parseNumber(number_chosen);
+            if (utils.anyIsError(number, number_chosen)) {
+                return error.value;
+            }
+            return exports.FACT(number) / (exports.FACT(number_chosen) * exports.FACT(number - number_chosen));
+        };
+    
+        exports.COMBINA = function(number, number_chosen) {
+            number = utils.parseNumber(number);
+            number_chosen = utils.parseNumber(number_chosen);
+            if (utils.anyIsError(number, number_chosen)) {
+                return error.value;
+            }
+            return (number === 0 && number_chosen === 0) ? 1 : exports.COMBIN(number + number_chosen - 1, number - 1);
+        };
+    
+        exports.COS = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.cos(number);
+        };
+    
+        exports.COSH = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return (Math.exp(number) + Math.exp(-number)) / 2;
+        };
+    
+        exports.COT = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return 1 / Math.tan(number);
+        };
+    
+        exports.COTH = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            var e2 = Math.exp(2 * number);
+            return (e2 + 1) / (e2 - 1);
+        };
+    
+        exports.CSC = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return 1 / Math.sin(number);
+        };
+    
+        exports.CSCH = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return 2 / (Math.exp(number) - Math.exp(-number));
+        };
+    
+        exports.DECIMAL = function(number, radix) {
+            if (arguments.length < 1) {
+                return error.value;
+            }
+    
+    
+            return parseInt(number, radix);
+        };
+    
+        exports.DEGREES = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return number * 180 / Math.PI;
+        };
+    
+        exports.EVEN = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return exports.CEILING(number, -2, -1);
+        };
+    
+        exports.EXP = Math.exp;
+    
+        var MEMOIZED_FACT = [];
+        exports.FACT = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            var n = Math.floor(number);
+            if (n === 0 || n === 1) {
+                return 1;
+            } else if (MEMOIZED_FACT[n] > 0) {
+                return MEMOIZED_FACT[n];
+            } else {
+                MEMOIZED_FACT[n] = exports.FACT(n - 1) * n;
+                return MEMOIZED_FACT[n];
+            }
+        };
+    
+        exports.FACTDOUBLE = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            var n = Math.floor(number);
+            if (n <= 0) {
+                return 1;
+            } else {
+                return n * exports.FACTDOUBLE(n - 2);
+            }
+        };
+    
+        exports.FLOOR = function(number, significance, mode) {
+            significance = (significance === undefined) ? 1 : significance;
+            mode = (mode === undefined) ? 0 : mode;
+    
+            number = utils.parseNumber(number);
+            significance = utils.parseNumber(significance);
+            mode = utils.parseNumber(mode);
+            if (utils.anyIsError(number, significance, mode)) {
+                return error.value;
+            }
+            if (significance === 0) {
+                return 0;
+            }
+    
+            significance = Math.abs(significance);
+            if (number >= 0) {
+                return Math.floor(number / significance) * significance;
+            } else {
+                if (mode === 0) {
+                    return -1 * Math.ceil(Math.abs(number) / significance) * significance;
+                } else {
+                    return -1 * Math.floor(Math.abs(number) / significance) * significance;
+                }
+            }
+        };
+    
+        exports.FLOOR.MATH = exports.FLOOR;
+    
+        exports.GCD = null;
+    
+        exports.INT = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.floor(number);
+        };
+    
+        exports.LCM = function() {
+            // Credits: Jonas Raoni Soares Silva
+            var o = utils.parseNumberArray(utils.flatten(arguments));
+            if (o instanceof Error) {
+                return o;
+            }
+            for (var i, j, n, d, r = 1;
+                (n = o.pop()) !== undefined;) {
+                while (n > 1) {
+                    if (n % 2) {
+                        for (i = 3, j = Math.floor(Math.sqrt(n)); i <= j && n % i; i += 2) {
+                          //empty
+                        }
+                        d = (i <= j) ? i : n;
+                    } else {
+                        d = 2;
+                    }
+                    for (n /= d, r *= d, i = o.length; i;
+                        (o[--i] % d) === 0 && (o[i] /= d) === 1 && o.splice(i, 1)) {
+                        //empty
+                    }
+                }
+            }
+            return r;
+        };
+    
+        exports.LN = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.log(number);
+        };
+    
+        exports.LOG = function(number, base) {
+            number = utils.parseNumber(number);
+            base = (base === undefined) ? 10 : utils.parseNumber(base);
+    
+            if (utils.anyIsError(number, base)) {
+                return error.value;
+            }
+    
+            return Math.log(number) / Math.log(base);
+        };
+    
+        exports.LOG10 = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.log(number) / Math.log(10);
+        };
+    
+        exports.MDETERM = null;
+    
+        exports.MINVERSE = null;
+    
+        exports.MMULT = null;
+    
+        exports.MOD = function(dividend, divisor) {
+            dividend = utils.parseNumber(dividend);
+            divisor = utils.parseNumber(divisor);
+            if (utils.anyIsError(dividend, divisor)) {
+                return error.value;
+            }
+            if (divisor === 0) {
+                return error.div0;
+            }
+            var modulus = Math.abs(dividend % divisor);
+            return (divisor > 0) ? modulus : -modulus;
+        };
+    
+        exports.MROUND = function(number, multiple) {
+            number = utils.parseNumber(number);
+            multiple = utils.parseNumber(multiple);
+            if (utils.anyIsError(number, multiple)) {
+                return error.value;
+            }
+            if (number * multiple < 0) {
+                return error.num;
+            }
+    
+            return Math.round(number / multiple) * multiple;
+        };
+    
+        exports.MULTINOMIAL = function() {
+            var args = utils.parseNumberArray(utils.flatten(arguments));
+            if (args instanceof Error) {
+                return args;
+            }
+            var sum = 0;
+            var divisor = 1;
+            for (var i = 0; i < args.length; i++) {
+                sum += args[i];
+                divisor *= exports.FACT(args[i]);
+            }
+            return exports.FACT(sum) / divisor;
+        };
+    
+        exports.MUNIT = null;
+    
+        exports.ODD = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            var temp = Math.ceil(Math.abs(number));
+            temp = (temp & 1) ? temp : temp + 1;
+            return (number > 0) ? temp : -temp;
+        };
+    
+        exports.PI = function() {
+            return Math.PI;
+        };
+    
+        exports.POWER = function(number, power) {
+            number = utils.parseNumber(number);
+            power = utils.parseNumber(power);
+            if (utils.anyIsError(number, power)) {
+                return error.value;
+            }
+            var result = Math.pow(number, power);
+            if (isNaN(result)) {
+                return error.num;
+            }
+    
+            return result;
+        };
+    
+        exports.PRODUCT = function() {
+            var args = utils.parseNumberArray(utils.flatten(arguments));
+            if (args instanceof Error) {
+                return args;
+            }
+            var result = 1;
+            for (var i = 0; i < args.length; i++) {
+                result *= args[i];
+            }
+            return result;
+        };
+    
+        exports.QUOTIENT = function(numerator, denominator) {
+            numerator = utils.parseNumber(numerator);
+            denominator = utils.parseNumber(denominator);
+            if (utils.anyIsError(numerator, denominator)) {
+                return error.value;
+            }
+            return parseInt(numerator / denominator, 10);
+        };
+    
+        exports.RADIANS = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return number * Math.PI / 180;
+        };
+    
+        exports.RAND = function() {
+            return Math.random();
+        };
+    
+        exports.RANDBETWEEN = function(bottom, top) {
+            bottom = utils.parseNumber(bottom);
+            top = utils.parseNumber(top);
+            if (utils.anyIsError(bottom, top)) {
+                return error.value;
+            }
+            // Creative Commons Attribution 3.0 License
+            // Copyright (c) 2012 eqcode
+            return bottom + Math.ceil((top - bottom + 1) * Math.random()) - 1;
+        };
+    
+        exports.ROMAN = null;
+    
+        exports.ROUND = function(number, digits) {
+            number = utils.parseNumber(number);
+            digits = utils.parseNumber(digits);
+            if (utils.anyIsError(number, digits)) {
+                return error.value;
+            }
+            return Math.round(number * Math.pow(10, digits)) / Math.pow(10, digits);
+        };
+    
+        exports.ROUNDDOWN = function(number, digits) {
+            number = utils.parseNumber(number);
+            digits = utils.parseNumber(digits);
+            if (utils.anyIsError(number, digits)) {
+                return error.value;
+            }
+            var sign = (number > 0) ? 1 : -1;
+            return sign * (Math.floor(Math.abs(number) * Math.pow(10, digits))) / Math.pow(10, digits);
+        };
+    
+        exports.ROUNDUP = function(number, digits) {
+            number = utils.parseNumber(number);
+            digits = utils.parseNumber(digits);
+            if (utils.anyIsError(number, digits)) {
+                return error.value;
+            }
+            var sign = (number > 0) ? 1 : -1;
+            return sign * (Math.ceil(Math.abs(number) * Math.pow(10, digits))) / Math.pow(10, digits);
+        };
+    
+        exports.SEC = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return 1 / Math.cos(number);
+        };
+    
+        exports.SECH = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return 2 / (Math.exp(number) + Math.exp(-number));
+        };
+    
+        exports.SERIESSUM = function(x, n, m, coefficients) {
+            x = utils.parseNumber(x);
+            n = utils.parseNumber(n);
+            m = utils.parseNumber(m);
+            coefficients = utils.parseNumberArray(coefficients);
+            if (utils.anyIsError(x, n, m, coefficients)) {
+                return error.value;
+            }
+            var result = coefficients[0] * Math.pow(x, n);
+            for (var i = 1; i < coefficients.length; i++) {
+                result += coefficients[i] * Math.pow(x, n + i * m);
+            }
+            return result;
+        };
+    
+        exports.SIGN = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            if (number < 0) {
+                return -1;
+            } else if (number === 0) {
+                return 0;
+            } else {
+                return 1;
+            }
+        };
+    
+        exports.SIN = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.sin(number);
+        };
+    
+        exports.SINH = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return (Math.exp(number) - Math.exp(-number)) / 2;
+        };
+    
+        exports.SQRT = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            if (number < 0) {
+                return error.num;
+            }
+            return Math.sqrt(number);
+        };
+    
+        exports.SQRTPI = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.sqrt(number * Math.PI);
+        };
+    
+        exports.SUBTOTAL = null;
+    
+        exports.ADD = function (num1, num2) {
+            if (arguments.length !== 2) {
+                return error.na;
+            }
+    
+            num1 = utils.parseNumber(num1);
+            num2 = utils.parseNumber(num2);
+            if (utils.anyIsError(num1, num2)) {
+                return error.value;
+            }
+    
+            return num1 + num2;
+        };
+    
+        exports.MINUS = function (num1, num2) {
+            if (arguments.length !== 2) {
+                return error.na;
+            }
+    
+            num1 = utils.parseNumber(num1);
+            num2 = utils.parseNumber(num2);
+            if (utils.anyIsError(num1, num2)) {
+                return error.value;
+            }
+    
+            return num1 - num2;
+        };
+    
+        exports.DIVIDE = function (dividend, divisor) {
+            if (arguments.length !== 2) {
+                return error.na;
+            }
+    
+            dividend = utils.parseNumber(dividend);
+            divisor = utils.parseNumber(divisor);
+            if (utils.anyIsError(dividend, divisor)) {
+                return error.value;
+            }
+    
+            if (divisor === 0) {
+                return error.div0;
+            }
+    
+            return dividend / divisor;
+        };
+    
+        exports.MULTIPLY = function (factor1, factor2) {
+            if (arguments.length !== 2) {
+                return error.na;
+            }
+    
+            factor1 = utils.parseNumber(factor1);
+            factor2 = utils.parseNumber(factor2);
+            if (utils.anyIsError(factor1, factor2)) {
+                return error.value;
+            }
+    
+            return factor1 * factor2;
+        };
+    
+        exports.GTE = function (num1, num2) {
+            if (arguments.length !== 2) {
+                return error.na;
+            }
+    
+            num1 = utils.parseNumber(num1);
+            num2 = utils.parseNumber(num2);
+            if (utils.anyIsError(num1, num2)) {
+                return error.error;
+            }
+    
+            return num1 >= num2;
+        };
+    
+        exports.LT = function (num1, num2) {
+            if (arguments.length !== 2) {
+                return error.na;
+            }
+    
+            num1 = utils.parseNumber(num1);
+            num2 = utils.parseNumber(num2);
+            if (utils.anyIsError(num1, num2)) {
+                return error.error;
+            }
+    
+            return num1 < num2;
+        };
+    
+        exports.LTE = function (num1, num2) {
+            if (arguments.length !== 2) {
+                return error.na;
+            }
+    
+            num1 = utils.parseNumber(num1);
+            num2 = utils.parseNumber(num2);
+            if (utils.anyIsError(num1, num2)) {
+                return error.error;
+            }
+    
+            return num1 <= num2;
+        };
+    
+        exports.EQ = function (value1, value2) {
+            if (arguments.length !== 2) {
+                return error.na;
+            }
+    
+            return value1 === value2;
+        };
+    
+        exports.NE = function (value1, value2) {
+            if (arguments.length !== 2) {
+                return error.na;
+            }
+    
+            return value1 !== value2;
+        };
+    
+        exports.POW = function (base, exponent) {
+            if (arguments.length !== 2) {
+                return error.na;
+            }
+    
+            base = utils.parseNumber(base);
+            exponent = utils.parseNumber(exponent);
+            if (utils.anyIsError(base, exponent)) {
+                return error.error;
+            }
+    
+            return exports.POWER(base, exponent);
+        };
+    
+        exports.SUM = function() {
+            var result = 0;
+            var argsKeys = Object.keys(arguments);
+            for (var i = 0; i < argsKeys.length; ++i) {
+                var elt = arguments[argsKeys[i]];
+                if (typeof elt === 'number') {
+                    result += elt;
+                } else if (typeof elt === 'string') {
+                    var parsed = parseFloat(elt);
+                    !isNaN(parsed) && (result += parsed);
+                } else if (Array.isArray(elt)) {
+                    result += exports.SUM.apply(null, elt);
+                }
+            }
+            return result;
+        };
+    
+        exports.SUMIF = function(range, criteria) {
+            range = utils.parseNumberArray(utils.flatten(range));
+            if (range instanceof Error) {
+                return range;
+            }
+            var result = 0;
+            for (var i = 0; i < range.length; i++) {
+                result += (eval(range[i] + criteria)) ? range[i] : 0; // jshint ignore:line
+            }
+            return result;
+        };
+    
+        exports.SUMIFS = function() {
+            var args = utils.argsToArray(arguments);
+            var range = utils.parseNumberArray(utils.flatten(args.shift()));
+            if (range instanceof Error) {
+                return range;
+            }
+            var criteria = args;
+    
+            var n_range_elements = range.length;
+            var n_criterias = criteria.length;
+    
+            var result = 0;
+            for (var i = 0; i < n_range_elements; i++) {
+                var el = range[i];
+                var condition = '';
+                for (var c = 0; c < n_criterias; c++) {
+                    condition += el + criteria[c];
+                    if (c !== n_criterias - 1) {
+                        condition += '&&';
+                    }
+                }
+                if (eval(condition)) { // jshint ignore:line
+                    result += el;
+                }
+            }
+            return result;
+        };
+    
+        exports.SUMPRODUCT = null;
+    
+        exports.SUMSQ = function() {
+            var numbers = utils.parseNumberArray(utils.flatten(arguments));
+            if (numbers instanceof Error) {
+                return numbers;
+            }
+            var result = 0;
+            var length = numbers.length;
+            for (var i = 0; i < length; i++) {
+                result += (ISNUMBER(numbers[i])) ? numbers[i] * numbers[i] : 0;
+            }
+            return result;
+        };
+    
+        exports.SUMX2MY2 = function(array_x, array_y) {
+            array_x = utils.parseNumberArray(utils.flatten(array_x));
+            array_y = utils.parseNumberArray(utils.flatten(array_y));
+            if (utils.anyIsError(array_x, array_y)) {
+                return error.value;
+            }
+            var result = 0;
+            for (var i = 0; i < array_x.length; i++) {
+                result += array_x[i] * array_x[i] - array_y[i] * array_y[i];
+            }
+            return result;
+        };
+    
+        exports.SUMX2PY2 = function(array_x, array_y) {
+            array_x = utils.parseNumberArray(utils.flatten(array_x));
+            array_y = utils.parseNumberArray(utils.flatten(array_y));
+            if (utils.anyIsError(array_x, array_y)) {
+                return error.value;
+            }
+            var result = 0;
+            array_x = utils.parseNumberArray(utils.flatten(array_x));
+            array_y = utils.parseNumberArray(utils.flatten(array_y));
+            for (var i = 0; i < array_x.length; i++) {
+                result += array_x[i] * array_x[i] + array_y[i] * array_y[i];
+            }
+            return result;
+        };
+    
+        exports.SUMXMY2 = function(array_x, array_y) {
+            array_x = utils.parseNumberArray(utils.flatten(array_x));
+            array_y = utils.parseNumberArray(utils.flatten(array_y));
+            if (utils.anyIsError(array_x, array_y)) {
+                return error.value;
+            }
+            var result = 0;
+            array_x = utils.flatten(array_x);
+            array_y = utils.flatten(array_y);
+            for (var i = 0; i < array_x.length; i++) {
+                result += Math.pow(array_x[i] - array_y[i], 2);
+            }
+            return result;
+        };
+    
+        exports.TAN = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return Math.tan(number);
+        };
+    
+        exports.TANH = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            var e2 = Math.exp(2 * number);
+            return (e2 - 1) / (e2 + 1);
+        };
+    
+        exports.TRUNC = function(number, digits) {
+            digits = (digits === undefined) ? 0 : digits;
+            number = utils.parseNumber(number);
+            digits = utils.parseNumber(digits);
+            if (utils.anyIsError(number, digits)) {
+                return error.value;
+            }
+            var sign = (number > 0) ? 1 : -1;
+            return sign * (Math.floor(Math.abs(number) * Math.pow(10, digits))) / Math.pow(10, digits);
+        };
+    
+        return exports;
+    })();
+    
+    jexcel.methods.misc = (function() {
+        var exports = {};
+    
+        exports.UNIQUE = function () {
+            var result = [];
+            for (var i = 0; i < arguments.length; ++i) {
+                var hasElement = false;
+                var element = arguments[i];
+    
+                // Check if we've already seen this element.
+                for (var j = 0; j < result.length; ++j) {
+                    hasElement = result[j] === element;
+                    if (hasElement) { break; }
+                }
+    
+                // If we did not find it, add it to the result.
+                if (!hasElement) {
+                    result.push(element);
+                }
+            }
+            return result;
+        };
+    
+        exports.FLATTEN = utils.flatten;
+    
+        exports.ARGS2ARRAY = function () {
+            return Array.prototype.slice.call(arguments, 0);
+        };
+    
+        exports.REFERENCE = function (context, reference) {
+            try {
+                var path = reference.split('.');
+                var result = context;
+                for (var i = 0; i < path.length; ++i) {
+                    var step = path[i];
+                    if (step[step.length - 1] === ']') {
+                        var opening = step.indexOf('[');
+                        var index = step.substring(opening + 1, step.length - 1);
+                        result = result[step.substring(0, opening)][index];
+                    } else {
+                        result = result[step];
+                    }
+                }
+                return result;
+            } catch (error) {}
+        };
+    
+        exports.JOIN = function (array, separator) {
+            return array.join(separator);
+        };
+    
+        exports.NUMBERS = function () {
+            var possibleNumbers = utils.flatten(arguments);
+            return possibleNumbers.filter(function (el) {
+                return typeof el === 'number';
+            });
+        };
+    
+        exports.NUMERAL = null;
+    
+        return exports;
+    })();
+    
+    jexcel.methods.text = (function() {
+        var exports = {};
+    
+        exports.ASC = null;
+    
+        exports.BAHTTEXT = null;
+    
+        exports.CHAR = function(number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return String.fromCharCode(number);
+        };
+    
+        exports.CLEAN = function(text) {
+            text = text || '';
+            var re = /[\0-\x1F]/g;
+            return text.replace(re, "");
+        };
+    
+        exports.CODE = function(text) {
+            text = text || '';
+            return text.charCodeAt(0);
+        };
+    
+        exports.CONCATENATE = function() {
+            var args = utils.flatten(arguments);
+    
+            var trueFound = 0;
+            while ((trueFound = args.indexOf(true)) > -1) {
+                args[trueFound] = 'TRUE';
+            }
+    
+            var falseFound = 0;
+            while ((falseFound = args.indexOf(false)) > -1) {
+                args[falseFound] = 'FALSE';
+            }
+    
+            return args.join('');
+        };
+    
+        exports.DBCS = null;
+    
+        exports.DOLLAR = null;
+    
+        exports.EXACT = function(text1, text2) {
+            return text1 === text2;
+        };
+    
+        exports.FIND = function(find_text, within_text, position) {
+            position = (position === undefined) ? 0 : position;
+            return within_text ? within_text.indexOf(find_text, position - 1) + 1 : null;
+        };
+    
+        exports.FIXED = null;
+    
+        exports.HTML2TEXT = function (value) {
+            var result = '';
+    
+            if (value) {
+                if (value instanceof Array) {
+                    value.forEach(function (line) {
+                        if (result !== '') {
+                          result += '\n';
+                        }
+                        result += (line.replace(/<(?:.|\n)*?>/gm, ''));
+                    });
+                } else {
+                    result = value.replace(/<(?:.|\n)*?>/gm, '');
+                }
+            }
+    
+            return result;
+        };
+    
+        exports.LEFT = function(text, number) {
+            number = (number === undefined) ? 1 : number;
+            number = utils.parseNumber(number);
+            if (number instanceof Error || typeof text !== 'string') {
+                return error.value;
+            }
+            return text ? text.substring(0, number) : null;
+        };
+    
+        exports.LEN = function(text) {
+            if (arguments.length === 0) {
+                return error.error;
+            }
+    
+            if (typeof text === 'string') {
+                return text ? text.length : 0;
+            }
+    
+            if (text.length) {
+                return text.length;
+            }
+    
+            return error.value;
+        };
+    
+        exports.LOWER = function(text) {
+            if (typeof text !== 'string') {
+                return error.value;
+            }
+            return text ? text.toLowerCase() : text;
+        };
+    
+        exports.MID = function(text, start, number) {
+            start = utils.parseNumber(start);
+            number = utils.parseNumber(number);
+            if (utils.anyIsError(start, number) || typeof text !== 'string') {
+                return number;
+            }
+    
+            var begin = start - 1;
+            var end = begin + number;
+    
+            return text.substring(begin, end);
+        };
+    
+        exports.NUMBERVALUE = null;
+    
+        exports.PRONETIC = null;
+    
+        exports.PROPER = function(text) {
+            if (text === undefined || text.length === 0) {
+                return error.value;
+            }
+            if (text === true) {
+                text = 'TRUE';
+            }
+            if (text === false) {
+                text = 'FALSE';
+            }
+            if (isNaN(text) && typeof text === 'number') {
+                return error.value;
+            }
+            if (typeof text === 'number') {
+                text = '' + text;
+            }
+    
+            return text.replace(/\w\S*/g, function(txt) {
+                return txt.charAt(0).toUpperCase() + txt.substr(1).toLowerCase();
+            });
+        };
+    
+        exports.REGEXEXTRACT = function (text, regular_expression) {
+            var match = text.match(new RegExp(regular_expression));
+            return match ? (match[match.length > 1 ? match.length - 1 : 0]) : null;
+        };
+    
+        exports.REGEXMATCH = function (text, regular_expression, full) {
+            var match = text.match(new RegExp(regular_expression));
+            return full ? match : !!match;
+        };
+    
+        exports.REGEXREPLACE = function (text, regular_expression, replacement) {
+            return text.replace(new RegExp(regular_expression), replacement);
+        };
+    
+        exports.REPLACE = function(text, position, length, new_text) {
+            position = utils.parseNumber(position);
+            length = utils.parseNumber(length);
+            if (utils.anyIsError(position, length) ||
+                typeof text !== 'string' ||
+                typeof new_text !== 'string') {
+                return error.value;
+            }
+            return text.substr(0, position - 1) + new_text + text.substr(position - 1 + length);
+        };
+    
+        exports.REPT = function(text, number) {
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return new Array(number + 1).join(text);
+        };
+    
+        exports.RIGHT = function(text, number) {
+            number = (number === undefined) ? 1 : number;
+            number = utils.parseNumber(number);
+            if (number instanceof Error) {
+                return number;
+            }
+            return text ? text.substring(text.length - number) : null;
+        };
+    
+        exports.SEARCH = function(find_text, within_text, position) {
+            var foundAt;
+            if (typeof find_text !== 'string' || typeof within_text !== 'string') {
+                return error.value;
+            }
+            position = (position === undefined) ? 0 : position;
+            foundAt = within_text.toLowerCase().indexOf(find_text.toLowerCase(), position - 1)+1;
+            return (foundAt === 0)?error.value:foundAt;
+        };
+    
+        exports.SPLIT = function (text, separator) {
+            return text.split(separator);
+        };
+    
+        exports.SUBSTITUTE = function(text, old_text, new_text, occurrence) {
+            if (!text || !old_text || !new_text) {
+                return text;
+            } else if (occurrence === undefined) {
+                return text.replace(new RegExp(old_text, 'g'), new_text);
+            } else {
+                var index = 0;
+                var i = 0;
+                while (text.indexOf(old_text, index) > 0) {
+                    index = text.indexOf(old_text, index + 1);
+                    i++;
+                    if (i === occurrence) {
+                        return text.substring(0, index) + new_text + text.substring(index + old_text.length);
+                    }
+                }
+            }
+        };
+    
+        exports.T = function(value) {
+            return (typeof value === "string") ? value : '';
+        };
+    
+        exports.TEXT = null;
+    
+        exports.TRIM = function(text) {
+            if (typeof text !== 'string') {
+                return error.value;
+            }
+            return text.replace(/ +/g, ' ').trim();
+        };
+    
+        exports.UNICHAR = exports.CHAR;
+    
+        exports.UNICODE = exports.CODE;
+    
+        exports.UPPER = function(text) {
+            if (typeof text !== 'string') {
+                return error.value;
+            }
+            return text.toUpperCase();
+        };
+    
+        exports.VALUE = null;
+    
+        return exports;
+    })();
+    
+    jexcel.methods.stats = (function() {
+        var exports = {};
+    
+        var SQRT2PI = 2.5066282746310002;
+    
+        exports.AVEDEV = null;
+    
+        exports.AVERAGE = function() {
+            var range = utils.numbers(utils.flatten(arguments));
+            var n = range.length;
+            var sum = 0;
+            var count = 0;
+            for (var i = 0; i < n; i++) {
+                sum += range[i];
+                count += 1;
+            }
+            return sum / count;
+        };
+    
+        exports.AVERAGEA = function() {
+            var range = utils.flatten(arguments);
+            var n = range.length;
+            var sum = 0;
+            var count = 0;
+            for (var i = 0; i < n; i++) {
+                var el = range[i];
+                if (typeof el === 'number') {
+                    sum += el;
+                }
+                if (el === true) {
+                    sum++;
+                }
+                if (el !== null) {
+                    count++;
+                }
+            }
+            return sum / count;
+        };
+    
+        exports.AVERAGEIF = function(range, criteria, average_range) {
+            average_range = average_range || range;
+            range = utils.flatten(range);
+            average_range = utils.parseNumberArray(utils.flatten(average_range));
+            if (average_range instanceof Error) {
+                return average_range;
+            }
+            var average_count = 0;
+            var result = 0;
+            for (var i = 0; i < range.length; i++) {
+                if (eval(range[i] + criteria)) { // jshint ignore:line
+                    result += average_range[i];
+                    average_count++;
+                }
+            }
+            return result / average_count;
+        };
+    
+        exports.AVERAGEIFS = null;
+    
+        exports.COUNT = function() {
+            return utils.numbers(utils.flatten(arguments)).length;
+        };
+    
+        exports.COUNTA = function() {
+            var range = utils.flatten(arguments);
+            return range.length - exports.COUNTBLANK(range);
+        };
+    
+        exports.COUNTIN = function (range, value) {
+            var result = 0;
+            for (var i = 0; i < range.length; i++) {
+                if (range[i] === value) {
+                    result++;
+                }
+            }
+            return result;
+        };
+    
+        exports.COUNTBLANK = function() {
+            var range = utils.flatten(arguments);
+            var blanks = 0;
+            var element;
+            for (var i = 0; i < range.length; i++) {
+                element = range[i];
+                if (element === null || element === '') {
+                    blanks++;
+                }
+            }
+            return blanks;
+        };
+    
+        exports.COUNTIF = function(range, criteria) {
+            range = utils.flatten(range);
+            if (!/[<>=!]/.test(criteria)) {
+                criteria = '=="' + criteria + '"';
+            }
+            var matches = 0;
+            for (var i = 0; i < range.length; i++) {
+                if (typeof range[i] !== 'string') {
+                    if (eval(range[i] + criteria)) { // jshint ignore:line
+                        matches++;
+                    }
+                } else {
+                    if (eval('"' + range[i] + '"' + criteria)) { // jshint ignore:line
+                        matches++;
+                    }
+                }
+            }
+            return matches;
+        };
+    
+        exports.COUNTIFS = function() {
+            var args = utils.argsToArray(arguments);
+            var results = new Array(utils.flatten(args[0]).length);
+            for (var i = 0; i < results.length; i++) {
+                results[i] = true;
+            }
+            for (i = 0; i < args.length; i += 2) {
+                var range = utils.flatten(args[i]);
+                var criteria = args[i + 1];
+                if (!/[<>=!]/.test(criteria)) {
+                    criteria = '=="' + criteria + '"';
+                }
+                for (var j = 0; j < range.length; j++) {
+                    if (typeof range[j] !== 'string') {
+                        results[j] = results[j] && eval(range[j] + criteria); // jshint ignore:line
+                    } else {
+                        results[j] = results[j] && eval('"' + range[j] + '"' + criteria); // jshint ignore:line
+                    }
+                }
+            }
+            var result = 0;
+            for (i = 0; i < results.length; i++) {
+                if (results[i]) {
+                    result++;
+                }
+            }
+            return result;
+        };
+    
+        exports.COUNTUNIQUE = function () {
+            return UNIQUE.apply(null, utils.flatten(arguments)).length;
+        };
+    
+        exports.FISHER = function(x) {
+            x = utils.parseNumber(x);
+            if (x instanceof Error) {
+                return x;
+            }
+            return Math.log((1 + x) / (1 - x)) / 2;
+        };
+    
+        exports.FISHERINV = function(y) {
+            y = utils.parseNumber(y);
+            if (y instanceof Error) {
+                return y;
+            }
+            var e2y = Math.exp(2 * y);
+            return (e2y - 1) / (e2y + 1);
+        };
+    
+        exports.FREQUENCY = function(data, bins) {
+            data = utils.parseNumberArray(utils.flatten(data));
+            bins = utils.parseNumberArray(utils.flatten(bins));
+            if (utils.anyIsError(data, bins)) {
+                return error.value;
+            }
+            var n = data.length;
+            var b = bins.length;
+            var r = [];
+            for (var i = 0; i <= b; i++) {
+                r[i] = 0;
+                for (var j = 0; j < n; j++) {
+                    if (i === 0) {
+                        if (data[j] <= bins[0]) {
+                            r[0] += 1;
+                        }
+                    } else if (i < b) {
+                        if (data[j] > bins[i - 1] && data[j] <= bins[i]) {
+                            r[i] += 1;
+                        }
+                    } else if (i === b) {
+                        if (data[j] > bins[b - 1]) {
+                            r[b] += 1;
+                        }
+                    }
+                }
+            }
+            return r;
+        };
+    
+        exports.LARGE = function(range, k) {
+            range = utils.parseNumberArray(utils.flatten(range));
+            k = utils.parseNumber(k);
+            if (utils.anyIsError(range, k)) {
+                return range;
+            }
+            return range.sort(function(a, b) {
+                return b - a;
+            })[k - 1];
+        };
+    
+        exports.MAX = function() {
+            var range = utils.numbers(utils.flatten(arguments));
+            return (range.length === 0) ? 0 : Math.max.apply(Math, range);
+        };
+    
+        exports.MAXA = function() {
+            var range = utils.arrayValuesToNumbers(utils.flatten(arguments));
+            return (range.length === 0) ? 0 : Math.max.apply(Math, range);
+        };
+    
+        exports.MIN = function() {
+            var range = utils.numbers(utils.flatten(arguments));
+            return (range.length === 0) ? 0 : Math.min.apply(Math, range);
+        };
+    
+        exports.MINA = function() {
+            var range = utils.arrayValuesToNumbers(utils.flatten(arguments));
+            return (range.length === 0) ? 0 : Math.min.apply(Math, range);
+        };
+    
+        exports.MODE = {};
+    
+        exports.MODE.MULT = function() {
+            // Credits: Roönaän
+            var range = utils.parseNumberArray(utils.flatten(arguments));
+            if (range instanceof Error) {
+                return range;
+            }
+            var n = range.length;
+            var count = {};
+            var maxItems = [];
+            var max = 0;
+            var currentItem;
+    
+            for (var i = 0; i < n; i++) {
+                currentItem = range[i];
+                count[currentItem] = count[currentItem] ? count[currentItem] + 1 : 1;
+                if (count[currentItem] > max) {
+                    max = count[currentItem];
+                    maxItems = [];
+                }
+                if (count[currentItem] === max) {
+                    maxItems[maxItems.length] = currentItem;
+                }
+            }
+            return maxItems;
+        };
+    
+        exports.MODE.SNGL = function() {
+            var range = utils.parseNumberArray(utils.flatten(arguments));
+            if (range instanceof Error) {
+                return range;
+            }
+            return exports.MODE.MULT(range).sort(function(a, b) {
+                return a - b;
+            })[0];
+        };
+    
+        exports.PERCENTILE = {};
+    
+        exports.PERCENTILE.EXC = function(array, k) {
+            array = utils.parseNumberArray(utils.flatten(array));
+            k = utils.parseNumber(k);
+            if (utils.anyIsError(array, k)) {
+                return error.value;
+            }
+            array = array.sort(function(a, b) {
+                {
+                    return a - b;
+                }
+            });
+            var n = array.length;
+            if (k < 1 / (n + 1) || k > 1 - 1 / (n + 1)) {
+                return error.num;
+            }
+            var l = k * (n + 1) - 1;
+            var fl = Math.floor(l);
+            return utils.cleanFloat((l === fl) ? array[l] : array[fl] + (l - fl) * (array[fl + 1] - array[fl]));
+        };
+    
+        exports.PERCENTILE.INC = function(array, k) {
+            array = utils.parseNumberArray(utils.flatten(array));
+            k = utils.parseNumber(k);
+            if (utils.anyIsError(array, k)) {
+                return error.value;
+            }
+            array = array.sort(function(a, b) {
+                return a - b;
+            });
+            var n = array.length;
+            var l = k * (n - 1);
+            var fl = Math.floor(l);
+            return utils.cleanFloat((l === fl) ? array[l] : array[fl] + (l - fl) * (array[fl + 1] - array[fl]));
+        };
+    
+        exports.PERCENTRANK = {};
+    
+        exports.PERCENTRANK.EXC = function(array, x, significance) {
+            significance = (significance === undefined) ? 3 : significance;
+            array = utils.parseNumberArray(utils.flatten(array));
+            x = utils.parseNumber(x);
+            significance = utils.parseNumber(significance);
+            if (utils.anyIsError(array, x, significance)) {
+                return error.value;
+            }
+            array = array.sort(function(a, b) {
+                return a - b;
+            });
+            var uniques = UNIQUE.apply(null, array);
+            var n = array.length;
+            var m = uniques.length;
+            var power = Math.pow(10, significance);
+            var result = 0;
+            var match = false;
+            var i = 0;
+            while (!match && i < m) {
+                if (x === uniques[i]) {
+                    result = (array.indexOf(uniques[i]) + 1) / (n + 1);
+                    match = true;
+                } else if (x >= uniques[i] && (x < uniques[i + 1] || i === m - 1)) {
+                    result = (array.indexOf(uniques[i]) + 1 + (x - uniques[i]) / (uniques[i + 1] - uniques[i])) / (n + 1);
+                    match = true;
+                }
+                i++;
+            }
+            return Math.floor(result * power) / power;
+        };
+    
+        exports.PERCENTRANK.INC = function(array, x, significance) {
+            significance = (significance === undefined) ? 3 : significance;
+            array = utils.parseNumberArray(utils.flatten(array));
+            x = utils.parseNumber(x);
+            significance = utils.parseNumber(significance);
+            if (utils.anyIsError(array, x, significance)) {
+                return error.value;
+            }
+            array = array.sort(function(a, b) {
+                return a - b;
+            });
+            var uniques = UNIQUE.apply(null, array);
+            var n = array.length;
+            var m = uniques.length;
+            var power = Math.pow(10, significance);
+            var result = 0;
+            var match = false;
+            var i = 0;
+            while (!match && i < m) {
+                if (x === uniques[i]) {
+                    result = array.indexOf(uniques[i]) / (n - 1);
+                    match = true;
+                } else if (x >= uniques[i] && (x < uniques[i + 1] || i === m - 1)) {
+                    result = (array.indexOf(uniques[i]) + (x - uniques[i]) / (uniques[i + 1] - uniques[i])) / (n - 1);
+                    match = true;
+                }
+                i++;
+            }
+            return Math.floor(result * power) / power;
+        };
+    
+        exports.PERMUT = function(number, number_chosen) {
+            number = utils.parseNumber(number);
+            number_chosen = utils.parseNumber(number_chosen);
+            if (utils.anyIsError(number, number_chosen)) {
+                return error.value;
+            }
+            return FACT(number) / FACT(number - number_chosen);
+        };
+    
+        exports.PERMUTATIONA = function(number, number_chosen) {
+            number = utils.parseNumber(number);
+            number_chosen = utils.parseNumber(number_chosen);
+            if (utils.anyIsError(number, number_chosen)) {
+                return error.value;
+            }
+            return Math.pow(number, number_chosen);
+        };
+    
+        exports.PHI = function(x) {
+            x = utils.parseNumber(x);
+            if (x instanceof Error) {
+                return error.value;
+            }
+            return Math.exp(-0.5 * x * x) / SQRT2PI;
+        };
+    
+        exports.PROB = function(range, probability, lower, upper) {
+            if (lower === undefined) {
+                return 0;
+            }
+            upper = (upper === undefined) ? lower : upper;
+    
+            range = utils.parseNumberArray(utils.flatten(range));
+            probability = utils.parseNumberArray(utils.flatten(probability));
+            lower = utils.parseNumber(lower);
+            upper = utils.parseNumber(upper);
+            if (utils.anyIsError(range, probability, lower, upper)) {
+                return error.value;
+            }
+    
+            if (lower === upper) {
+                return (range.indexOf(lower) >= 0) ? probability[range.indexOf(lower)] : 0;
+            }
+    
+            var sorted = range.sort(function(a, b) {
+                return a - b;
+            });
+            var n = sorted.length;
+            var result = 0;
+            for (var i = 0; i < n; i++) {
+                if (sorted[i] >= lower && sorted[i] <= upper) {
+                    result += probability[range.indexOf(sorted[i])];
+                }
+            }
+            return result;
+        };
+    
+        exports.QUARTILE = {};
+    
+        exports.QUARTILE.EXC = function(range, quart) {
+            range = utils.parseNumberArray(utils.flatten(range));
+            quart = utils.parseNumber(quart);
+            if (utils.anyIsError(range, quart)) {
+                return error.value;
+            }
+            switch (quart) {
+                case 1:
+                    return exports.PERCENTILE.EXC(range, 0.25);
+                case 2:
+                    return exports.PERCENTILE.EXC(range, 0.5);
+                case 3:
+                    return exports.PERCENTILE.EXC(range, 0.75);
+                default:
+                    return error.num;
+            }
+        };
+    
+        exports.QUARTILE.INC = function(range, quart) {
+            range = utils.parseNumberArray(utils.flatten(range));
+            quart = utils.parseNumber(quart);
+            if (utils.anyIsError(range, quart)) {
+                return error.value;
+            }
+            switch (quart) {
+                case 1:
+                    return exports.PERCENTILE.INC(range, 0.25);
+                case 2:
+                    return exports.PERCENTILE.INC(range, 0.5);
+                case 3:
+                    return exports.PERCENTILE.INC(range, 0.75);
+                default:
+                    return error.num;
+            }
+        };
+    
+        exports.RANK = {};
+    
+        exports.RANK.AVG = function(number, range, order) {
+            number = utils.parseNumber(number);
+            range = utils.parseNumberArray(utils.flatten(range));
+            if (utils.anyIsError(number, range)) {
+                return error.value;
+            }
+            range = utils.flatten(range);
+            order = order || false;
+            var sort = (order) ? function(a, b) {
+                return a - b;
+            } : function(a, b) {
+                return b - a;
+            };
+            range = range.sort(sort);
+    
+            var length = range.length;
+            var count = 0;
+            for (var i = 0; i < length; i++) {
+                if (range[i] === number) {
+                    count++;
+                }
+            }
+    
+            return (count > 1) ? (2 * range.indexOf(number) + count + 1) / 2 : range.indexOf(number) + 1;
+        };
+    
+        exports.RANK.EQ = function(number, range, order) {
+            number = utils.parseNumber(number);
+            range = utils.parseNumberArray(utils.flatten(range));
+            if (utils.anyIsError(number, range)) {
+                return error.value;
+            }
+            order = order || false;
+            var sort = (order) ? function(a, b) {
+                return a - b;
+            } : function(a, b) {
+                return b - a;
+            };
+            range = range.sort(sort);
+            return range.indexOf(number) + 1;
+        };
+    
+        exports.RSQ = function(data_x, data_y) { // no need to flatten here, PEARSON will take care of that
+            data_x = utils.parseNumberArray(utils.flatten(data_x));
+            data_y = utils.parseNumberArray(utils.flatten(data_y));
+            if (utils.anyIsError(data_x, data_y)) {
+                return error.value;
+            }
+            return Math.pow(exports.PEARSON(data_x, data_y), 2);
+        };
+    
+        exports.SMALL = function(range, k) {
+            range = utils.parseNumberArray(utils.flatten(range));
+            k = utils.parseNumber(k);
+            if (utils.anyIsError(range, k)) {
+                return range;
+            }
+            return range.sort(function(a, b) {
+                return a - b;
+            })[k - 1];
+        };
+    
+        exports.STANDARDIZE = function(x, mean, sd) {
+            x = utils.parseNumber(x);
+            mean = utils.parseNumber(mean);
+            sd = utils.parseNumber(sd);
+            if (utils.anyIsError(x, mean, sd)) {
+                return error.value;
+            }
+            return (x - mean) / sd;
+        };
+    
+        exports.STDEV = {};
+    
+        exports.STDEV.P = function() {
+            var v = exports.VAR.P.apply(this, arguments);
+            return Math.sqrt(v);
+        };
+    
+        exports.STDEV.S = function() {
+            var v = exports.VAR.S.apply(this, arguments);
+            return Math.sqrt(v);
+        };
+    
+        exports.STDEVA = function() {
+            var v = exports.VARA.apply(this, arguments);
+            return Math.sqrt(v);
+        };
+    
+        exports.STDEVPA = function() {
+            var v = exports.VARPA.apply(this, arguments);
+            return Math.sqrt(v);
+        };
+    
+        exports.VAR = {};
+    
+        exports.VAR.P = function() {
+            var range = utils.numbers(utils.flatten(arguments));
+            var n = range.length;
+            var sigma = 0;
+            var mean = exports.AVERAGE(range);
+            for (var i = 0; i < n; i++) {
+                sigma += Math.pow(range[i] - mean, 2);
+            }
+            return sigma / n;
+        };
+    
+        exports.VAR.S = function() {
+            var range = utils.numbers(utils.flatten(arguments));
+            var n = range.length;
+            var sigma = 0;
+            var mean = exports.AVERAGE(range);
+            for (var i = 0; i < n; i++) {
+                sigma += Math.pow(range[i] - mean, 2);
+            }
+            return sigma / (n - 1);
+        };
+    
+        exports.VARA = function() {
+            var range = utils.flatten(arguments);
+            var n = range.length;
+            var sigma = 0;
+            var count = 0;
+            var mean = exports.AVERAGEA(range);
+            for (var i = 0; i < n; i++) {
+                var el = range[i];
+                if (typeof el === 'number') {
+                    sigma += Math.pow(el - mean, 2);
+                } else if (el === true) {
+                    sigma += Math.pow(1 - mean, 2);
+                } else {
+                    sigma += Math.pow(0 - mean, 2);
+                }
+    
+                if (el !== null) {
+                    count++;
+                }
+            }
+            return sigma / (count - 1);
+        };
+    
+        exports.VARPA = function() {
+            var range = utils.flatten(arguments);
+            var n = range.length;
+            var sigma = 0;
+            var count = 0;
+            var mean = exports.AVERAGEA(range);
+            for (var i = 0; i < n; i++) {
+                var el = range[i];
+                if (typeof el === 'number') {
+                    sigma += Math.pow(el - mean, 2);
+                } else if (el === true) {
+                    sigma += Math.pow(1 - mean, 2);
+                } else {
+                    sigma += Math.pow(0 - mean, 2);
+                }
+    
+                if (el !== null) {
+                    count++;
+                }
+            }
+            return sigma / count;
+        };
+    
+        exports.WEIBULL = {};
+    
+        exports.WEIBULL.DIST = function(x, alpha, beta, cumulative) {
+            x = utils.parseNumber(x);
+            alpha = utils.parseNumber(alpha);
+            beta = utils.parseNumber(beta);
+            if (utils.anyIsError(x, alpha, beta)) {
+                return error.value;
+            }
+            return (cumulative) ? 1 - Math.exp(-Math.pow(x / beta, alpha)) : Math.pow(x, alpha - 1) * Math.exp(-Math.pow(x / beta, alpha)) * alpha / Math.pow(beta, alpha);
+        };
+    
+        exports.Z = {};
+    
+        exports.Z.TEST = function(range, x, sd) {
+            range = utils.parseNumberArray(utils.flatten(range));
+            x = utils.parseNumber(x);
+            if (utils.anyIsError(range, x)) {
+                return error.value;
+            }
+    
+            sd = sd || exports.STDEV.S(range);
+            var n = range.length;
+            return 1 - exports.NORM.S.DIST((exports.AVERAGE(range) - x) / (sd / Math.sqrt(n)), true);
+        };
+    
+        return exports;
+    })();
+
+    jexcel.methods.utils = (function() {
+        var exports = {};
+
+        exports.PROGRESS = function(p, c) {
+            var color = c ? c : 'red';
+            var value = p ? p : '0';
+
+            return '<div style="width:' + value + '%;height:4px;background-color:' + color + ';margin-top:1px;"></div>';
+        };
+
+        exports.RATING = function(v) {
+            var html = '<div class="jrating">';
+            for (var i = 0; i < 5; i++) {
+                if (i < v) {
+                    html += '<div class="jrating-selected"></div>';
+                } else {
+                    html += '<div></div>';
+                }
+            }
+            html += '</div>';
+            return html;
+        }
+
+        return exports;
+    })();
+
+    for (var i = 0; i < Object.keys(jexcel.methods).length; i++) {
+        var methods = jexcel.methods[Object.keys(jexcel.methods)[i]];
+        for (var j = 0; j < Object.keys(methods).length; j++) {
+            if (typeof(methods[Object.keys(methods)[j]]) == 'function') {
+                window[Object.keys(methods)[j]] = methods[Object.keys(methods)[j]];
+            } else {
+                window[Object.keys(methods)[j]] = function() {
+                    return Object.keys(methods)[j] + 'Not implemented';
+                }
+            }
+        }
+    }
+
+    return jexcel;
+})));
\ No newline at end of file
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.js.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.js.xml
new file mode 100644
index 0000000000000000000000000000000000000000..06094cf5020b3e189c97faa82754f0e17ead3a13
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jexcel/jexcel.js.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>jexcel.js</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>application/javascript</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites.xml
new file mode 100644
index 0000000000000000000000000000000000000000..fdafdb7edba4d238f6cd265924036864a5692587
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites.xml
@@ -0,0 +1,26 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="Folder" module="OFS.Folder"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>_objects</string> </key>
+            <value>
+              <tuple/>
+            </value>
+        </item>
+        <item>
+            <key> <string>id</string> </key>
+            <value> <string>jsuites</string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.css.css b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.css.css
new file mode 100644
index 0000000000000000000000000000000000000000..2d5f8b871f4b2fad5bd46d34a2cc8d400c9e4902
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.css.css
@@ -0,0 +1,3274 @@
+
+/**
+ * (c) jSuites Javascript Web Components
+ *
+ * Author: Paul Hodel <paul.hodel@gmail.com>
+ * Website: https://bossanova.uk/jsuites/
+ * Description: Create amazing web based applications.
+ *
+ * MIT License
+ *
+ */
+
+/** General **/
+
+:root {
+    --button-color: #298BA8; 
+    --active-color: #007aff;
+}
+
+.jdragging {
+    opacity:0.2;
+    filter: alpha(opacity=20);
+}
+
+.jupload {
+    background-image: url();
+    background-repeat: no-repeat;
+    background-size: 100px;
+    background-position: center;
+    background-color: rgba(230, 230, 230, 0.1);
+    border: 1px dotted #eee;
+    cursor: pointer;
+    box-sizing: border-box;
+    width:100%;
+    min-height:180px;
+}
+
+.jremove {
+    opacity: 0.2;
+    filter: alpha(opacity=20);
+}
+
+/** Animations **/
+.fade-in {
+    animation: fade-in 2s forwards;
+}
+
+.fade-out {
+    animation: fade-out 1s forwards;
+}
+
+.slide-left-in {
+    animation: slide-left-in 0.4s forwards;
+}
+
+.slide-left-out {
+    animation: slide-left-out 0.4s forwards;
+}
+
+.slide-right-in {
+    animation: slide-right-in 0.4s forwards;
+}
+
+.slide-right-out {
+    animation: slide-right-out 0.4s forwards;
+}
+
+.slide-top-in {
+    animation: slide-top-in 0.4s forwards;
+}
+
+.slide-top-out {
+    animation: slide-top-out 0.2s forwards;
+}
+
+.slide-bottom-in {
+    animation: slide-bottom-in 0.4s forwards;
+}
+
+.slide-bottom-out {
+    animation: slide-bottom-out 0.1s forwards;
+}
+
+/** Fadein and Fadeout **/
+@keyframes fade-in {
+    0% { opacity: 0; }
+    100% { opacity: 100; }
+}
+
+@-webkit-keyframes fade-in {
+    0% { opacity: 0; }
+    100% { opacity: 100; }
+}
+
+@keyframes fade-out {
+    0% { opacity: 100; }
+    100% { opacity: 0; }
+}
+
+@-webkit-keyframes fade-out {
+    0% { opacity: 100; }
+    100% { opacity: 0; }
+}
+
+/** Keyframes Left to Right **/
+@keyframes slide-left-in {
+    0% { transform: translateX(-100%); }
+    100% { transform: translateX(0%); }
+}
+
+@-webkit-keyframes slide-left-in {
+    0% { transform: translateX(-100%); }
+    100% { -webkit-transform: translateX(0%); }
+}
+    
+@keyframes slide-left-out {
+    0% { transform: translateX(0%); }
+    100% { transform: translateX(-100%); }
+}
+
+@-webkit-keyframes slide-left-out {
+    0% { -webkit-transform: translateX(0%); }
+    100% { -webkit-transform: translateX(-100%); }
+}
+
+/** Keyframes Right to Left **/
+@keyframes slide-right-in {
+    0% { transform: translateX(100%); }
+    100% { transform: translateX(0%); }
+}
+
+@-webkit-keyframes slide-right-in
+{
+    0% { transform: translateX(100%); }
+    100% { -webkit-transform: translateX(0%); }
+}
+    
+@keyframes slide-right-out {
+    0% { transform: translateX(0%); }
+    100% { transform: translateX(100%); }
+}
+
+@-webkit-keyframes slide-right-out {
+    0% { -webkit-transform: translateX(0%); }
+    100% { -webkit-transform: translateX(100%); }
+}
+
+/** Keyframes Top to Bottom **/
+@keyframes slide-top-in {
+    0% { transform: translateY(-100%); }
+    100% { transform: translateY(0%); }
+}
+
+@-webkit-keyframes slide-top-in {
+    0% { transform: translateY(-100%); }
+    100% { -webkit-transform: translateY(0%); }
+}
+    
+@keyframes slide-top-out {
+    0% { transform: translateY(0%); }
+    100% { transform: translateY(-100%); }
+}
+
+@-webkit-keyframes slide-top-out {
+    0% { -webkit-transform: translateY(0%); }
+    100% { -webkit-transform: translateY(-100%); }
+}
+
+/** Keyframes Bottom to Top **/
+@keyframes slide-bottom-in {
+    0% { transform: translateY(100%); }
+    100% { transform: translateY(0%); }
+}
+
+@-webkit-keyframes slide-bottom-in {
+    0% { transform: translateY(100%); }
+    100% { -webkit-transform: translateY(0%); }
+}
+    
+@keyframes slide-bottom-out {
+    0% { transform: translateY(0%); }
+    100% { transform: translateY(100%); }
+}
+
+@-webkit-keyframes slide-bottom-out {
+    0% { -webkit-transform: translateY(0%); }
+    100% { -webkit-transform: translateY(100%); }
+}
+
+
+@supports (-webkit-overflow-scrolling: touch) {
+    .app .options input:checked:before {
+        top:-12px;
+    }
+}
+
+@-webkit-keyframes spin {
+    from {
+        -webkit-transform:rotate(0deg);
+    }
+    to {
+        -webkit-transform:rotate(360deg);
+    }
+}
+
+@keyframes spin {
+    from {
+        transform:rotate(0deg);
+    }
+    to {
+        transform:rotate(360deg);
+    }
+}
+
+.unselectable {
+    -webkit-touch-callout: none;
+    -webkit-user-select: none;
+    -khtml-user-select: none;
+    -moz-user-select: none;
+    -ms-user-select: none;
+    user-select: none;
+}
+
+.jbutton {
+    padding:4px;
+    padding-left:10px;
+    padding-right:20px;
+    border:1px solid #ddd;
+    display:inline-flex;
+    border-radius:4px;
+    cursor:pointer;
+    line-height: 24px;
+    -webkit-font-smoothing: antialiased;
+}
+
+.jbutton.blue {
+    border:1px solid transparent;
+    color:#fff;
+    background-color:#1a73e8;
+}
+
+.jbutton::before {
+    content: attr(data-icon);
+    width: 24px;
+    height: 24px;
+    font-size: 24px;
+    line-height: 24px;
+    font-family: 'Material icons';
+    color: #999;
+    margin-right:2px;
+}
+
+.jbutton.blue::before {
+    color: #fff;
+}
+
+.jbutton:hover {
+    background-color: #eee;
+}
+
+.jbutton.blue:hover {
+    background-color: #1a73e8;
+    opacity: 0.8;
+}
+
+/**
+ * Date & Datetime picker v1.0.1
+ * Author: paul.hodel@gmail.com
+ * https://github.com/paulhodel/jtools
+ */
+ 
+.jcalendar {
+    position:relative;
+    z-index:9000;
+    display:none;
+    box-sizing:border-box;
+    -webkit-touch-callout: none;
+    -webkit-user-select: none;
+    -khtml-user-select: none;
+    -moz-user-select: none;
+    -ms-user-select: none;
+    user-select: none;
+    -webkit-tap-highlight-color: rgba(0,0,0,0);
+    -webkit-tap-highlight-color: transparent;
+    min-width:280px;
+}
+
+.jcalendar-focus {
+    display:block;
+}
+
+.jcalendar-backdrop {
+    position:fixed;
+    top:0px;
+    left:0px;
+    z-index:9000;
+    min-width:100%;
+    min-height:100%;
+    background-color:rgba(0,0,0,0.5);
+    border:0px;
+    padding:0px;
+    display:none;
+}
+
+.jcalendar-container {
+    position:relative;
+    box-sizing:border-box;
+}
+
+.jcalendar-content {
+    position:absolute;
+    z-index:9001;
+    -webkit-box-shadow: 1px 1px 5px 0px rgba(0,0,0,0.39);
+    -moz-box-shadow: 1px 1px 5px 0px rgba(0,0,0,0.39);
+    box-shadow: 1px 1px 5px 0px rgba(0,0,0,0.39);
+    background-color:#fff;
+}
+
+.jcalendar-table > table {
+    width:100%;
+    background-color:#fff;
+}
+
+.jcalendar-table > table > tbody td {
+    box-sizing:border-box;
+    cursor:pointer;
+    padding:9px;
+    font-size:0.9em;
+}
+
+.jcalendar-table > table > thead {
+    cursor:pointer;
+}
+
+.jcalendar-header {
+    text-align:center;
+}
+
+.jcalendar-header span {
+    margin-right:4px;
+    font-size:1.1em;
+    font-weight:bold;
+}
+
+.jcalendar-prev {
+    cursor:pointer;
+    background-image:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M15.41 16.59L10.83 12l4.58-4.59L14 6l-6 6 6 6 1.41-1.41z' fill='%23000' /%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3C/svg%3E");
+    background-position:center;
+    background-repeat:no-repeat;
+}
+
+.jcalendar-next {
+    cursor:pointer;
+    background-image:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M8.59 16.59L13.17 12 8.59 7.41 10 6l6 6-6 6-1.41-1.41z' fill='%23000' /%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3C/svg%3E");
+    background-position:center;
+    background-repeat:no-repeat;
+}
+
+.jcalendar-weekday {
+    font-weight: 600;
+    background-color: #fcfcfc;
+    padding: 14px;
+}
+
+.jcalendar thead td {
+    padding:10px;
+    height:40px;
+}
+
+.jcalendar tfoot td {
+    padding:10px;
+}
+
+.jcalendar-months td, .jcalendar-years td {
+    height:24px;
+} 
+
+.jcalendar-input {
+    padding-right:18px;
+    background-image:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' viewBox='0 0 24 24' fill='gray'%3E%3Cpath d='M20 3h-1V1h-2v2H7V1H5v2H4c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 18H4V8h16v13z'/%3E%3Cpath fill='none' d='M0 0h24v24H0z'/%3E%3C/svg%3E");
+    background-position:top 50% right 5px;
+    background-repeat:no-repeat;
+    box-sizing: border-box;
+}
+
+.jcalendar-done {
+    -webkit-box-shadow: 1px 1px 5px 0px rgba(0,0,0,0.39);
+    -moz-box-shadow: 1px 1px 5px 0px rgba(0,0,0,0.39);
+    box-shadow: 1px 1px 5px 0px rgba(0,0,0,0.39);
+    background-color:#fff;
+}
+
+.jcalendar-update {
+    border:1px solid #ccc;
+    background-color:#fff;
+    border-radius:4px;
+    padding:5px;
+    width:100%;
+}
+
+.jcalendar select {
+    width:55px;
+    display:inline-block;
+    border:0px;
+    padding:4px;
+    text-align:center;
+    font-size:1.1em;
+    user-select:none;
+    margin-right:10px;
+}
+
+.jcalendar select:first-child
+{
+    margin-right:2px;
+}
+
+.jcalendar-selected {
+    background-color:#eee;
+}
+
+.jcalendar-reset, .jcalendar-confirm {
+    text-transform:uppercase;
+    cursor:pointer;
+    color: var(--active-color);
+}
+
+.jcalendar-controls {
+    padding:15px;
+
+    -webkit-box-sizing: border-box;
+    box-sizing: border-box;
+    vertical-align:middle;
+
+    display: -webkit-box;
+    display: -moz-box;
+    display: -ms-flexbox;
+    display: -webkit-flex;
+    display: flex;
+
+    -webkit-flex-flow: row wrap;
+    justify-content: space-between;
+    align-items:center;
+
+    border-bottom:1px solid #ddd;
+}
+
+.jcalendar-controls div {
+    font-weight:bold;
+}
+
+.jcalendar-fullsize  {
+    position:fixed;
+    width:100%;
+    top:0px;
+    left:0px;
+}
+
+.jcalendar-fullsize .jcalendar-content
+{
+    position:fixed;
+    width:100%;
+    left:0px;
+    bottom:0px;
+}
+
+.jcalendar-focus.jcalendar-fullsize .jcalendar-backdrop {
+    display:block;
+}
+
+.jcalendar-sunday {
+    color: red;
+}
+.jcalendar-disabled {
+    color: #ccc;
+}
+
+.jcolor {
+    position: absolute;
+    display: none;
+    outline: none;
+}
+
+.jcolor-content {
+    position: absolute;
+    left: 0px;
+    z-index: 9000;
+    user-select: none;
+    -webkit-font-smoothing: antialiased;
+    font-size: .875rem;
+    letter-spacing: .2px;
+    -webkit-border-radius: 4px;
+    border-radius: 4px;
+    -webkit-box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2);
+    box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2);
+    background-color:#fff;
+    box-sizing: border-box;
+}
+
+.jcolor-content table {
+    padding: 7px;
+    box-sizing: border-box;
+}
+
+.jcolor-focus {
+    display:block;
+}
+
+.jcolor table {
+    width:100%;
+    height:100%;
+}
+
+.jcolor td {
+    border:2px solid #fff;
+}
+
+.jcolor-selected {
+    border:2px solid #000 !important;
+    background-repeat:no-repeat;
+    background-size: cover;
+    background-position:0 0;
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3Cpath d='M9 16.17L4.83 12l-1.42 1.41L9 19 21 7l-1.41-1.41z' fill='white'/%3E%3C/svg%3E");
+}
+
+.jcolor-fullscreen {
+    position: fixed;
+    bottom: 0px;
+    width:100%;
+    max-height:250px;
+    border-radius: 0px;
+    box-sizing: border-box;
+}
+
+.jcolor-close {
+    display: none;
+    font-size: 1em;
+    color: var(--active-color);
+    text-transform: uppercase;
+    text-align: right;
+    padding: 15px;
+    font-weight: bold;
+    -webkit-box-shadow: 1px 0px 1px 0px rgba(0,0,0,0.39);
+    -moz-box-shadow: 1px 0px 1px 0px rgba(0,0,0,0.39);
+    box-shadow: 1px 0px 1px 0px rgba(0,0,0,0.39);
+    width: 100%;
+    box-sizing: border-box;
+}
+
+.jcolor-fullscreen .jcolor-close {
+    display: block;
+}
+
+.jcolor-backdrop {
+    position: fixed;
+    top: 0px;
+    left: 0px;
+    min-width: 100%;
+    min-height: 100%;
+    background-color: rgba(0,0,0,0.5);
+    border: 0px;
+    padding: 0px;
+    z-index: 8000;
+    display: none;
+    
+  -webkit-touch-callout: none; /* iOS Safari */
+    -webkit-user-select: none; /* Safari */
+     -khtml-user-select: none; /* Konqueror HTML */
+       -moz-user-select: none; /* Firefox */
+        -ms-user-select: none; /* Internet Explorer/Edge */
+            user-select: none; /* Non-prefixed version, currently
+                                  supported by Chrome and Opera */
+}
+
+/**
+ * Contextmenu v1.0.1
+ * Author: paul.hodel@gmail.com
+ * https://github.com/paulhodel/jsuites
+ */
+ 
+.jcontextmenu {
+    position:fixed;
+    z-index:10000;
+    background:#fff;
+    color: #555;
+    font-size: 11px;
+    -webkit-user-select: none;
+    -moz-user-select: none;
+    user-select: none;
+    -webkit-box-shadow: 2px 2px 2px 0px rgba(143, 144, 145, 1);
+    -moz-box-shadow: 2px 2px 2px 0px rgba(143, 144, 145, 1);
+    box-shadow: 2px 2px 2px 0px rgba(143, 144, 145, 1);
+    border: 1px solid #C6C6C6;
+    padding: 0px;
+    padding-top:4px;
+    padding-bottom:4px;
+    margin:0px;
+    outline:none;
+    display:none;
+}
+
+.jcontextmenu.jcontextmenu-focus {
+    display:inline-block;
+}
+
+.jcontextmenu > div {
+    box-sizing: border-box;
+    display: block;
+    padding: 8px 8px 8px 28px;
+    width: 250px;
+    position: relative;
+    cursor: default;
+    font-size: 11px;
+    font-family:sans-serif;
+}
+
+.jcontextmenu > div a {
+    color: #555;
+    text-decoration: none;
+}
+
+.jcontextmenu > div span {
+    float: right;
+    margin-right:10px;
+}
+
+.jcontextmenu .contextmenu-disabled {
+    color: #a1a192;
+}
+
+.jcontextmenu > div:not(.contextmenu-line):hover {
+    background: #ebebeb;
+}
+
+.jcontextmenu > div.contextmenu-line {
+    border-top: 1px solid #e9e9e9;
+    margin-top:5px;
+    padding:2px;
+}
+
+.jcontextmenu hr {
+    border: 1px solid #e9e9e9;
+    border-bottom: 0;
+    margin-top:5px;
+    margin-bottom:5px;
+}
+
+/**
+ * Dialog v1.0.1
+ * Author: paul.hodel@gmail.com
+ * https://github.com/paulhodel/jtools
+ */
+ 
+.jdialog
+{
+    position:fixed;
+    left: 0;
+    top: 0;
+    width: 100%;
+    min-height:100%;
+    z-index:10000;
+
+    background-color:rgba(0,0,0,0.5);
+    -webkit-transition-duration: .1s;
+    transition-duration: .1s;
+    display: flex;
+    -ms-flex-align: center;
+    -webkit-align-items: center;
+    -webkit-box-align: center;
+
+    align-items: center;
+}
+
+.jdialog .jdialog-container
+{
+    width:100%;
+    max-width:280px;
+    box-sizing: border-box;
+    margin:0 auto;
+    vertical-align:middle;
+    background-color:#fff;
+    border-radius:10px;
+
+    opacity: 0;
+    transition: opacity .2s;
+    -webkit-transition: opacity .2s;
+}
+
+.jdialog .jdialog-header
+{
+    padding:10px;
+    text-align:center;
+}
+
+.jdialog .jdialog-header .jdialog-title
+{
+    font-size:1em;
+    font-weight:bold;
+    padding:8px;
+}
+
+.jdialog .jdialog-header .jdialog-message
+{
+    padding: 4px;
+}
+
+.jdialog .jdialog-footer
+{
+    border-top: 1px solid #ccc;
+    display: flex;
+    flex-wrap: wrap;
+    align-content: stretch;
+    align-items: center;
+    text-align: center;
+}
+
+.jdialog .jdialog-footer > div
+{
+    padding:4px;
+    border-left:1px solid #ccc;
+    text-align:center;
+    width:100%;
+}
+.jdialog .jdialog-footer > div:first-child
+{
+    border-left:0px;
+}
+
+.jdialog .jdialog-footer input
+{
+    border: 0px;
+    margin: 0px;
+    background-color: transparent;
+    color: var(--active-color);
+    width: 100%;
+    outline: none;
+}
+
+/**
+ * (c) 2013 jDropdown
+ * http://www.github.com/paulhodel/jdropdown
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Custom dropdowns
+ */
+
+.jdropdown
+{
+    cursor:pointer;
+    -webkit-touch-callout: none;
+    -webkit-user-select: none;
+    -khtml-user-select: none;
+    -moz-user-select: none;
+    -ms-user-select: none;
+    user-select: none;
+    box-sizing: border-box;
+    background:#fff;
+    -webkit-tap-highlight-color: transparent;
+    display: inline-block;
+}
+
+.jdropdown-header::placeholder
+{
+    color:#000;
+}
+
+.jdropdown-backdrop
+{
+    position:fixed;
+    top:0px;
+    left:0px;
+    min-width:100%;
+    min-height:100%;
+    background-color:rgba(0,0,0,0.5);
+    border:0px;
+    padding:0px;
+    z-index:8000;
+    display:none;
+}
+
+.jdropdown-focus
+{
+    position:relative;
+}
+
+.jdropdown-focus .jdropdown-container
+{
+    transform: translate3d(0,0,0);
+}
+
+.jdropdown-focus .jdropdown-header
+{
+    outline:auto 5px -webkit-focus-ring-color;
+}
+
+.jdropdown-container-header
+{
+    padding:0px;
+    margin:0px;
+    position:relative;
+}
+
+.jdropdown-header
+{
+    width:100%;
+    appearance: none;
+    background-repeat: no-repeat;
+    background-position:top 50% right 5px;
+    background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3Cpath d='M7 10l5 5 5-5H7z' fill='gray'/%3E%3C/svg%3E");
+    text-overflow: ellipsis;
+    cursor:pointer;
+    box-sizing: border-box;
+    -webkit-appearance: none;
+    -moz-appearance: none;
+    padding-right:24px;
+}
+
+.jdropdown-insert-button
+{
+    font-size: 1.4em;
+    text-transform: uppercase;
+    position:absolute;
+    right: 30px;
+    top: 4px;
+    display:none;
+}
+
+.jdropdown-container
+{
+    min-width: inherit;
+    transform: translate3d(-10000px,0,0);
+    position:absolute;
+    z-index:9001;
+}
+
+.jdropdown-close
+{
+    display:none;
+    font-size:1em;
+    color: var(--active-color);
+    text-transform:uppercase;
+    text-align:right;
+    padding:15px;
+    font-weight:bold;
+}
+
+.jdropdown-content
+{
+    min-width:inherit;
+    margin:0px;
+    box-sizing:border-box;
+}
+
+.jdropdown-content:empty
+{
+}
+
+.jdropdown-item
+{
+    white-space: nowrap;
+    text-align: left;
+    text-overflow: ellipsis;
+    overflow-x: hidden;
+    color: #000;
+    display: flex;
+    align-items: center;
+}
+
+.jdropdown-image
+{
+    margin-right:10px;
+    width: 32px;
+    height: 32px;
+    border-radius:20px;
+}
+
+.jdropdown-image-small
+{
+    width:24px;
+    height:24px;
+}
+
+.jdropdown-title
+{
+    font-size: 0.7em;
+    text-overflow: ellipsis;
+    overflow-x: hidden;
+    display: block;
+}
+
+/** Default visual **/
+
+.jdropdown-default .jdropdown-header
+{
+    border:1px solid #ccc;
+    padding:5px;
+    padding-left:10px;
+    padding-right:16px;
+}
+
+.jdropdown-default .jdropdown-container
+{
+    background-color:#fff;
+}
+
+.jdropdown-default.jdropdown-focus.jdropdown-insert .jdropdown-header {
+    padding-right:50px;
+}
+
+.jdropdown-default.jdropdown-focus.jdropdown-insert .jdropdown-insert-button {
+    display:block;
+}
+
+.jdropdown-default .jdropdown-content
+{
+    min-width:inherit;
+    border:1px solid #8fb1e3;
+    margin:0px;
+    background-color:#fff;
+    box-sizing:border-box;
+    min-height:10px;
+    max-height:215px;
+    overflow-y:auto;
+}
+
+.jdropdown-default .jdropdown-item
+{
+    padding:4px;
+    padding-left:8px;
+    padding-right:40px;
+}
+
+.jdropdown-default .jdropdown-item:hover
+{
+    background-color:#1f93ff;
+    color:#fff;
+}
+
+.jdropdown-default .jdropdown-cursor
+{
+    background-color:#1f93ff;
+    color:#fff;
+}
+
+.jdropdown-default .jdropdown-selected
+{
+    background-image: url('');
+    background-repeat:no-repeat;
+    background-position:top 50% right 5px;
+    background-color:#1f93ff;
+    color:#fff;
+}
+
+.jdropdown-default .jdropdown-group
+{
+    margin-top:5px;
+}
+
+.jdropdown-default .jdropdown-group .jdropdown-item
+{
+    padding-left:16px;
+}
+
+.jdropdown-default .jdropdown-group-name
+{
+    padding-left:8px;
+    font-weight:bold;
+}
+
+
+/** Default render for mobile **/
+
+.jdropdown-picker.jdropdown-focus .jdropdown-header {
+    outline: none;
+}
+
+.jdropdown-picker.jdropdown-focus .jdropdown-backdrop {
+    display:block;
+}
+
+.jdropdown-picker .jdropdown-container
+{
+    position:fixed;
+    bottom:0px;
+    left:0px;
+    border-bottom:1px solid #e6e6e8;
+    width:100%;
+    background-color:#fff;
+    box-sizing: border-box;
+}
+
+.jdropdown-picker .jdropdown-close
+{
+    -webkit-box-shadow: 0px -1px 5px 0px rgba(0,0,0,0.39);
+    -moz-box-shadow: 0px -1px 5px 0px rgba(0,0,0,0.39);
+    box-shadow: 0px -1px 5px 0px rgba(0,0,0,0.39);
+    background-color:#fff;
+    display:block;
+}
+
+.jdropdown-picker .jdropdown-content
+{
+    overflow-y:scroll;
+    height:280px;
+    background-color:#fafafa;
+    border-top:1px solid #e6e6e8;
+}
+
+.jdropdown-picker .jdropdown-group-name
+{
+    font-size: 1em;
+    text-transform: uppercase;
+    padding-top:10px;
+    padding-bottom:10px;
+    display: block;
+    border-bottom: 1px solid #e6e6e8;
+    padding-left:20px;
+    padding-right:20px;
+    text-align:center;
+    font-weight:bold;
+}
+
+.jdropdown-picker .jdropdown-item
+{
+    font-size: 1em;
+    text-transform: uppercase;
+    padding-top:10px;
+    padding-bottom:10px;
+    border-bottom: 1px solid #e6e6e8;
+    padding-left:20px;
+    padding-right:20px;
+}
+
+.jdropdown-picker .jdropdown-selected
+{
+    background-image: url('');
+    background-repeat:no-repeat;
+    background-position:top 50% right 15px;
+    background-color:#1f93ff;
+    color:#fff;
+}
+
+.jdropdown-picker .jdropdown-cursor
+{
+    background-color:#1f93ff;
+    color:#fff;
+}
+
+/** Default render for mobile searchbar **/
+
+.jdropdown-searchbar.jdropdown-focus
+{
+    position:fixed;
+    top:0px;
+    left:0px;
+    width:100%;
+    height:100%;
+    background-color:#fafafa;
+    padding:0px;
+    z-index:9001;
+    overflow-y:scroll;
+}
+
+.jdropdown-searchbar.jdropdown-focus .jdropdown-container-header
+{
+    padding:10px;
+    background-color:#fff;
+    box-shadow: 0 1px 2px rgba(0,0,0,.1);
+}
+
+.jdropdown-searchbar.jdropdown-focus .jdropdown-header
+{
+    border: 0px;
+    background-repeat: no-repeat;
+    background-position-x: 0%;
+    background-position-y: 40%;
+    background-image: url();
+    padding-left: 30px !important;
+    padding-right: 60px !important;
+    outline: none;
+}
+
+.jdropdown-searchbar.jdropdown-focus .jdropdown-close
+{
+    display:block;
+}
+
+.jdropdown-searchbar .jdropdown-container
+{
+    width:100%;
+}
+
+.jdropdown-searchbar .jdropdown-close
+{
+    position:fixed;
+    top:0px;
+    right:0px;
+}
+
+.jdropdown-searchbar .jdropdown-content
+{
+    margin-top:10px;
+}
+
+.jdropdown-searchbar .jdropdown-group
+{
+    margin-top:10px;
+    margin-bottom:15px;
+    background-color:#fff;
+}
+
+.jdropdown-searchbar .jdropdown-group-name
+{
+    border-top: 1px solid #e6e6e8;
+    border-bottom: 1px solid #e6e6e8;
+    padding:10px;
+    padding-left:12px;
+    font-weight:bold;
+}
+
+.jdropdown-searchbar .jdropdown-group-arrow
+{
+    float:right;
+    width:24px;
+    height:24px;
+    background-repeat:no-repeat;
+}
+
+.jdropdown-searchbar .jdropdown-group-arrow-down
+{
+    background-image: url();
+}
+
+.jdropdown-searchbar .jdropdown-group-arrow-up
+{
+    background-image: url();
+}
+
+.jdropdown-searchbar .jdropdown-item
+{
+    padding-top:10px;
+    padding-bottom:10px;
+    border-bottom: 1px solid #e6e6e8;
+    padding-left:15px;
+    padding-right:40px;
+    background-color:#fff;
+    font-size:0.9em;
+}
+
+.jdropdown-searchbar .jdropdown-description {
+    text-overflow: ellipsis;
+    overflow: hidden;
+    max-width: calc(100% - 20px);
+}
+
+.jdropdown-searchbar .jdropdown-content > .jdropdown-item:first-child
+{
+    border-top: 1px solid #e6e6e8;
+}
+
+.jdropdown-searchbar .jdropdown-selected
+{
+    background-image: url('');
+    background-repeat:no-repeat;
+    background-position:top 50% right 15px;
+}
+
+/** List render **/
+
+.jdropdown-list
+{
+}
+
+.jdropdown-list .jdropdown-container
+{
+    display:block;
+}
+
+.jdropdown-list .jdropdown-header
+{
+    display:none;
+}
+
+.jdropdown-list .jdropdown-group
+{
+    background-color:#fff;
+}
+
+.jdropdown-list .jdropdown-group-name
+{
+    border-bottom: 1px solid #e6e6e8;
+    padding-top:10px;
+    padding-bottom:10px;
+    font-weight:bold;
+}
+
+.jdropdown-list .jdropdown-item
+{
+    padding-top:10px;
+    padding-bottom:10px;
+    border-bottom: 1px solid #e6e6e8;
+    padding-left:10px;
+    padding-right:40px;
+    background-color:#fff;
+}
+
+.jdropdown-list .jdropdown-selected
+{
+    background-image: url('');
+    background-repeat:no-repeat;
+    background-position:top 50% right 10px;
+}
+
+@media only screen and (max-device-width : 800px)
+{
+    .jdropdown-list
+    {
+        width:100% !important;
+        border:0px;
+        padding:0px;
+    }
+
+    .jdropdown-list .jdropdown-container
+    {
+        min-width:100%;
+    }
+}
+
+.app .jdropdown-item {
+    text-transform:uppercase;
+}
+
+/**
+ * (c) jTools Text Editor
+ * https://github.com/paulhodel/jtools
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Inline richtext editor
+ */
+
+ .jeditor-container {
+    border:1px solid #ccc;
+}
+
+.jeditor-dragging {
+     border:1px dashed #000;
+}
+
+.jeditor-container.jeditor-padding {
+    padding:10px;
+}
+
+.jeditor {
+    outline:none;
+    word-break: break-word;
+
+}
+
+.jeditor-container.jeditor-padding .jeditor {
+    min-height:120px;
+    margin-bottom:10px;
+    padding:10px;
+}
+
+.jeditor-toolbar {
+    cursor: pointer;
+    display: -webkit-inline-box;
+    display: -webkit-inline-flex;
+    display: -ms-inline-flexbox;
+    display: inline-flex;
+    -webkit-align-items: center;
+    align-items: center;
+    border: none;
+    -webkit-border-radius: 2px;
+    border-radius: 2px;
+    -webkit-box-shadow: 0 4px 5px 0 rgba(0,0,0,0.14), 0 1px 10px 0 rgba(0,0,0,0.12), 0 2px 4px -1px rgba(0,0,0,0.2);
+    box-shadow: 0 4px 5px 0 rgba(0,0,0,0.14), 0 1px 10px 0 rgba(0,0,0,0.12), 0 2px 4px -1px rgba(0,0,0,0.2);
+    margin: 0px;
+    margin-bottom:5px;
+    padding: 5px;
+    white-space: nowrap;
+    border-radius:2px;
+}
+
+.jeditor-toolbar .jeditor-toolbar-group {
+    background-color:#f3f3f3;
+}
+
+.jeditor-toolbar div {
+    padding:4px;
+}
+
+.jeditor-toolbar div.jeditor-toolbar-button {
+    padding-left:20px;
+    padding-right:20px;
+}
+
+.jeditor-toolbar div {
+    -webkit-user-select: none;  /* Chrome all / Safari all */
+    -moz-user-select: none;     /* Firefox all */
+    -ms-user-select: none;      /* IE 10+ */
+    user-select: none;          /* Likely future */   
+}
+
+.jeditor-toolbar div i {
+    display:block;
+    font-size:20px;
+    width:20px;
+    height:20px;
+    line-height:20px;
+    color:#777;
+}
+
+.jeditor-toolbar select {
+    border:0px;
+    background-color:transparent;
+    padding:0px;
+    padding-left:5px;
+    padding-right:5px;
+    display:block;
+    height:32px;
+    outline:none;
+    border-radius:4px;
+    margin:1px;
+}
+
+.jeditor-toolbar .jtoolbar-divisor {
+    width: 2px;
+    height: 28px;
+    padding: 0px;
+    margin: 4px;
+    background-color: #f2f2f2;
+}
+
+.jeditor-toolbar select option {
+    background-color:#fff;
+    padding:20px;
+}
+
+.jeditor-toolbar div:hover, .toolbar select:hover {
+    background-color:#f2f2f2;
+    border-radius:2px;
+}
+
+.jeditor-toolbar div:hover i {
+    color:#222;
+}
+
+.jeditor-toolbar div.selected {
+    background-color:#cecece;
+}
+
+.jeditor-thumbs {
+    display:inline-block;
+    margin-right:5px;
+}
+
+.jeditor-thumbs-container {
+    margin-top:15px;
+}
+
+.jeditor-thumbs .close {
+    position:absolute;
+    height:1px;
+}
+
+.jeditor-thumbs .close i {
+    color:#fff;
+    text-shadow: 0px 0px 2px #000;
+    position:relative;
+    top:5px;
+    left:72px;
+}
+
+.jeditor-thumbs img {
+    border:1px solid #eee;
+    width:100px;
+    height:100px;
+}
+
+.jeditor-users {
+    position:absolute;
+    border:1px solid #ccc;
+    background-color:#fff;
+    padding-top:5px;
+    padding-bottom:5px;
+    max-height:220px;
+    overflow-y:scroll;
+}
+
+.jeditor-users > div
+{
+    display:flex;
+    align-items:center;
+    min-width:220px;
+    padding:10px;
+    padding-top:5px;
+    padding-bottom:5px;
+}
+
+.jeditor-users > div:hover
+{
+    background-color:#efefef;
+}
+
+.jeditor-users > div img
+{
+    width:24px;
+    height:24px;
+    border-radius:12px;
+    margin-right:5px;
+}
+
+/** Snippet **/
+
+.snippet {
+    margin-top:15px;
+    cursor:pointer;
+    border: 1px solid #eee;
+    position:relative;
+}
+
+.snippet:focus {
+    outline: none;
+}
+
+.snippet img {
+    width:25%;
+    max-width:120px;
+    float:left;
+    margin-right:10px;
+    margin-bottom:10px;
+}
+
+.snippet .snippet-title {
+    margin-top:15px;
+    font-size:1.4em;
+}
+
+.snippet .snippet-description {
+    margin-top:5px;
+    font-size:1em;
+}
+
+.snippet .snippet-host {
+    margin-top:10px;
+    margin-bottom:10px;
+    text-transform:uppercase;
+    font-size:0.8em;
+    color:#777;
+    text-align:right;
+}
+
+.snippet .snippet-url {
+    display:none;
+}
+
+.jeditor .snippet:after {
+    content:'';
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E");
+    position:absolute;
+    top:0;
+    right:0;
+    margin:14px;
+    font-size:24px;
+    width:24px;
+    height:24px;
+    cursor:pointer;
+    text-shadow: 0px 0px 5px #fff;
+}
+
+.snippet div, .snippet img {
+    -webkit-user-select: none;
+    -khtml-user-select: none;
+    -moz-user-select: none;
+    -o-user-select: none;
+    user-select: none;
+
+    -webkit-user-drag: none;
+    -khtml-user-drag: none;
+    -moz-user-drag: none;
+    -o-user-drag: none;
+}
+
+.jeditor img {
+    border:2px solid transparent;
+    box-sizing: border-box;
+}
+
+.jeditor img:focus {
+    border:2px solid #0096FD;
+}
+
+.jeditor .pdf {
+    background-image: url("data:image/svg+xml,%3Csvg version='1.1' id='Layer_1' xmlns='http://www.w3.org/2000/svg' xmlns:xlink='http://www.w3.org/1999/xlink' x='0px' y='0px' viewBox='0 0 512 512' style='enable-background:new 0 0 512 512;' xml:space='preserve'%3E%3Cpath style='fill:%23C30B15;' d='M511.344,274.266C511.77,268.231,512,262.143,512,256C512,114.615,397.385,0,256,0S0,114.615,0,256 c0,117.769,79.53,216.949,187.809,246.801L511.344,274.266z'/%3E%3Cpath style='fill:%2385080E;' d='M511.344,274.266L314.991,77.913L119.096,434.087l68.714,68.714C209.522,508.787,232.385,512,256,512 C391.243,512,501.976,407.125,511.344,274.266z'/%3E%3Cpolygon style='fill:%23FFFFFF;' points='278.328,333.913 255.711,77.913 119.096,77.913 119.096,311.652 '/%3E%3Cpolygon style='fill:%23E8E6E6;' points='392.904,311.652 392.904,155.826 337.252,133.565 314.991,77.913 255.711,77.913 256.067,333.913 '/%3E%3Cpolygon style='fill:%23FFFFFF;' points='314.991,155.826 314.991,77.913 392.904,155.826 '/%3E%3Crect x='119.096' y='311.652' style='fill:%23FC0F1A;' width='273.809' height='122.435'/%3E%3Cg%3E%3Cpath style='fill:%23FFFFFF;' d='M204.871,346.387c13.547,0,21.341,6.659,21.341,18.465c0,12.412-7.795,19.601-21.341,19.601h-9.611 v14.909h-13.471v-52.975L204.871,346.387L204.871,346.387z M195.26,373.858h8.93c5.904,0,9.308-2.952,9.308-8.552 c0-5.525-3.406-8.324-9.308-8.324h-8.93V373.858z'/%3E%3Cpath style='fill:%23FFFFFF;' d='M257.928,346.387c16.649,0,28.152,10.746,28.152,26.487c0,15.666-11.655,26.488-28.683,26.488 h-22.25v-52.975H257.928z M248.619,388.615h9.611c8.249,0,14.151-6.357,14.151-15.665c0-9.384-6.205-15.817-14.757-15.817h-9.006 V388.615z'/%3E%3Cpath style='fill:%23FFFFFF;' d='M308.563,356.982v12.26h23.763v10.596h-23.763v19.525h-13.471v-52.975h39.277v10.595h-25.806 V356.982z'/%3E%3C/g%3E%3C/svg%3E%0A");
+    background-repeat: no-repeat;
+    background-size: cover;
+    width:60px;
+    height:60px;
+}
+
+
+
+/**
+ * (c) jLoading
+ * https://github.com/paulhodel/jtools
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Page loading spin
+ */
+
+.jloading {
+    position:fixed;
+    z-index:10001;
+    width:100%;
+    left:0;
+    right:0;
+    top:0;
+    bottom:0;
+    background-color: rgba(0,0,0,0.7);
+}
+
+.jloading::after {
+    content:'';
+    display:block;
+    margin:0 auto;
+    margin-top:50vh;
+    width:40px;
+    height:40px;
+    border-style:solid;
+    border-color:white;
+    border-top-color:transparent;
+    border-width:4px;
+    border-radius:50%;
+    -webkit-animation: spin .8s linear infinite;
+    animation: spin .8s linear infinite;
+}
+
+.jloading.spin {
+    background-color:transparent;
+}
+
+.jloading.spin::after {
+    margin:0 auto;
+    margin-top:80px;
+    border-color:#aaa;
+    border-top-color:transparent;
+}
+
+
+.jlogin {
+    width: 100%;
+    box-sizing: border-box;
+    margin:0 auto;
+    vertical-align:middle;
+    border-radius: 5px;
+    -webkit-border-radius: 5px;
+    -moz-border-radius: 5px;
+    font-size:1em;
+}
+
+.jlogin-fullscreen {
+    position:absolute;
+    top:0px;
+    left:0px;
+    width:100%;
+    height:100%;
+    background-color:#fff;
+    z-index:9000;
+}
+
+.jlogin > form {
+    max-width:480px;
+    padding:20px;
+    margin: 0 auto;
+}
+
+.jlogin > form > div {
+    margin-bottom:10px;
+}
+
+.jlogin label {
+    display:block;
+}
+
+.jlogin > form > div > input {
+    width:100%;
+    outline:none;
+    padding:15px;
+    margin:0px;
+}
+
+.jlogin input[type='checkbox'] {
+    float:left;
+}
+
+.jlogin input[type='button'] {
+    padding:10px;
+    background-color: var(--button-color);
+    border: 1px solid var(--button-color);
+    color: #fff;
+    cursor:pointer;
+    -webkit-appearance: none;
+    -moz-appearance: none;
+    appearance: none;
+}
+
+.jlogin span
+{
+    margin:0 auto;
+}
+.jlogin img
+{
+    text-align:center;
+    max-width:220px;
+}
+
+.jlogin .jlogin-logo
+{
+    text-align:center;
+    padding:20px;
+}
+
+.jlogin .captcha
+{
+    width:100%;
+    margin-top:4px;
+    margin-bottom:4px;
+    border:1px solid #ccc;
+    display:block;
+}
+
+.jlogin .facebookButton
+{
+    padding:10px;
+    background-color: var(--button-color);
+    border: 1px solid var(--button-color);
+    color: #fff;
+    text-align:center;
+    background-repeat:no-repeat;
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' width='24px' height='24px'%3E%3Cpath d='M19,3H5C3.895,3,3,3.895,3,5v14c0,1.105,0.895,2,2,2h7.621v-6.961h-2.343v-2.725h2.343V9.309 c0-2.324,1.421-3.591,3.495-3.591c0.699-0.002,1.397,0.034,2.092,0.105v2.43h-1.428c-1.13,0-1.35,0.534-1.35,1.322v1.735h2.7 l-0.351,2.725h-2.365V21H19c1.105,0,2-0.895,2-2V5C21,3.895,20.105,3,19,3z' fill='white'/%3E%3C/svg%3E%0A");
+    background-position:10px 40%;
+    cursor:pointer;
+}
+
+.jlogin .rememberButton
+{
+    padding:10px;
+    display:none;
+}
+
+.jlogin .requestButton, .jlogin .cancelButton, .jlogin .newProfileButton
+{
+    padding:20px;
+    padding-bottom:0px;
+    text-align:center;
+    cursor:pointer;
+}
+
+.jlogin-captcha img {
+    min-width:280px;
+}
+
+.jlogin-loading:before {
+    content: "";
+    display: block;
+    position: fixed;
+    top: 0px;
+    left: 0px;
+    width: 200px;
+    height: 4px;
+    background-color: var(--button-color);
+    animation: loading 2s linear infinite;
+}
+
+@keyframes loading {
+    from { left: 0px; width: 25%; }
+    25%  { width: 25%; }
+    50%  { width: 50%; }
+    75%  { left:  75%; }
+    100% { left:  100%; }
+    to { left: 100%; }
+}
+
+@media only screen and (max-device-width : 800px)
+{
+    .jsuites * {
+        -webkit-tap-highlight-color: transparent;
+        -webkit-touch-callout: none;
+    }
+
+    html.jsuites {
+        height:100%;
+    }
+
+    .jsuites body {
+        margin:0px;
+        padding:0px;
+        width:100%;
+        height:100%;
+        min-height:100%;
+        line-height:1.4;
+        font-family:-apple-system,SF UI Text,Helvetica Neue,Helvetica,Arial,sans-serif !important;
+        background-color:#fff;
+    }
+
+    .jsuites input, .jsuites select, .jsuites textarea, .jsuites div {
+        font-family:-apple-system,SF UI Text,Helvetica Neue,Helvetica,Arial,sans-serif !important;
+    }
+}
+
+.japp
+{
+    height:100%;
+    box-sizing: border-box;
+    overflow:scroll;
+    background-color:#fafafa;
+}
+
+.jwarning::before {
+    content:'No internet connection';
+    background-color:red;
+    color:#fff;
+    padding:2px;
+    position:fixed;
+    top:0px;
+    left:0px;
+    z-index:20000;
+    width:100%;
+    font-size:0.55em;
+    text-align:center;
+}
+
+/*
+.japp .activeHighLight
+{
+    color:var(--active-color);
+}
+
+.japp .active a, .japp .active span, .japp .active i
+{
+    color:var(--active-color);
+}
+
+.japp .no-padding
+{
+    padding:0px;
+}
+
+.japp .action
+{
+    color:var(--active-color);
+    cursor:pointer;
+}
+
+
+.japp i.action
+{
+    position:absolute;
+    right:0px;
+    margin-right:15px;
+    cursor:pointer;
+}
+*/
+
+.japp button {
+    border-radius: 4px;
+    width: 100%;
+    border: 1px solid var(--active-color);
+    color: var(--active-color);
+    background-color: transparent;
+    padding: 4px;
+    outline: none;
+}
+
+.japp button:active {
+    border: 1px solid #000;
+    color: #000;
+}
+
+.japp .red {
+    color: red;
+    border-color: 1px solid red;
+}
+
+.japp .link {
+    color: var(--active-color);
+    cursor: pointer;
+}
+
+.japp .uppercase {
+    text-transform: uppercase;
+}
+
+/** Navbar **/
+
+.japp .navbar {
+    position:fixed;
+    top: 0;
+    width: 100%;
+    z-index: 500;
+    margin: 0;
+    -webkit-backface-visibility: hidden;
+    -webkit-transform: translate3d(0,0,0);
+    -webkit-box-sizing: border-box;
+    backface-visibility: hidden;
+    box-sizing: border-box;
+    transform: translate3d(0,0,0);
+    height: 44px;
+    background-color: #fff;
+    box-shadow: 0 1px 2px rgba(0,0,0,.1);
+}
+
+.jwarning .navbar {
+    top:16px;
+}
+
+.japp .navbar-container {
+    padding: 0 8px;
+    position: absolute;
+    left: 0;
+    top: 0;
+    width: 100%;
+    height: 100%;
+    -webkit-box-sizing: border-box;
+    box-sizing: border-box;
+    vertical-align:middle;
+
+    display: flex;
+
+    -webkit-flex-flow: row wrap;
+    justify-content: space-between;
+    align-items:center;
+}
+
+.japp .navbar-container div {
+    text-align:center;
+}
+
+.japp .navbar-container div.title {
+    display: flex;
+}
+
+.japp .navbar-container div.title div
+{
+    line-height:24px;
+}
+
+.japp .navbar-container div.icon {
+    width:24px;
+}
+
+.japp .navbar-container div i {
+    display:block;
+}
+
+.japp .navbar-container div img {
+    height:28px;
+    display:block;
+}
+
+.japp .navbar-container div.title div img {
+    border-radius:12px;
+    width:24px;
+    height:24px;
+    margin-right:6px;
+}
+
+.japp .navbar .icon {
+    color:var(--active-color);
+}
+
+.japp .title {
+    font-size:1.2em;
+}
+
+.japp .block {
+    padding:15px;
+    color: #6d6d72;
+    font-size:0.9em;
+    box-sizing: border-box;
+}
+
+.japp .block-title {
+    text-transform: uppercase;
+    color: #6d6d72;
+    margin: 25px 15px 10px;
+    line-height: 17px;
+    position: relative;
+    overflow: hidden;
+    white-space: nowrap;
+    text-overflow: ellipsis;
+    line-height: 1;
+    font-size:0.9em;
+}
+
+.japp .block-strong {
+    background-color:#fff;
+    color: #000;
+}
+
+.japp .block-border {
+    border-top:1px solid #ddd;
+    border-bottom:1px solid #ddd; 
+}
+
+.japp .block-footer {
+	text-transform: uppercase;
+    padding:15px;
+    color: #6d6d72;
+    font-size:0.7em;
+}
+
+.japp .block-collapse {
+    max-height:100px;
+    overflow-y:hidden;
+}
+
+.japp .block-instruction {
+    text-transform: uppercase;
+    padding:15px;
+    color: #6d6d72;
+    font-size:0.7em;
+    text-align:center;
+}
+
+.japp .pages {
+    display: flex;
+    flex-wrap: nowrap;
+    align-items: flex-start;
+    height: 100%;
+}
+
+.japp .page {
+    padding-top: 45px;
+    padding-bottom: 45px;
+    min-height: 100%;
+    min-width: 100%;
+    box-sizing: border-box;
+    overflow-y: auto;
+    /*will-change: scroll-position;
+    -webkit-overflow-scrolling: touch;
+    -webkit-transform: translateZ(0px);
+    -webkit-transform: translate3d(0,0,0);
+    -webkit-perspective: 1000;*/
+}
+
+.warning .page {
+    padding-top: 60px;
+}
+
+.japp .progressbar-container {
+    margin: 15px;
+    margin-left: 0px;
+    margin-right: 0px;
+    margin-top: 5px;
+}
+
+.japp .progressbar {
+    width:100%;
+    border:1px solid var(--active-color);
+    border-radius:1px;
+}
+
+.japp .progressbar-title {
+    margin-bottom:2px;
+    text-transform:uppercase;
+    display:flex;
+    justify-content:space-between;
+}
+
+.japp .progressbar div {
+    background-color:var(--active-color);
+    height:4px;
+    width:0%;
+}
+
+.japp .panel {
+    position:fixed;
+    top: 0;
+    left: 0;
+    margin: 0;
+    width:60vw;
+    max-width:220px;
+    height:100vh;
+    background: #fff;
+    z-index: 1003;
+    -webkit-box-sizing: border-box;
+    box-sizing: border-box;
+    box-shadow: 0 -1px 2px rgba(0,0,0,.1);
+    padding:4px;
+}
+
+.japp .panel-left {
+    -webkit-box-shadow: 0px 2px 15px -5px rgba(0, 0, 0, 0.7);
+    box-shadow: 0px 2px 15px -5px rgba(0, 0, 0, 0.7);
+}
+
+.japp .panel a {
+    color:var(--active-color);
+}
+
+/** Toolbar **/
+
+.japp .jtoolbar
+{
+    position:fixed;
+    bottom: 0;
+    margin: 0;
+    left: 0;
+    width: 100%;
+    background: #f7f7f8;
+    z-index: 500;
+    -webkit-box-sizing: border-box;
+    box-sizing: border-box;
+    box-shadow: 0 -1px 2px rgba(0,0,0,.1);
+    padding:4px;
+}
+
+.japp .jtoolbar > div
+{
+    -webkit-box-sizing: border-box;
+    box-sizing: border-box;
+    vertical-align:middle;
+
+    display: -webkit-box;
+    display: -moz-box;
+    display: -ms-flexbox;
+    display: -webkit-flex;
+    display: flex;
+
+    -webkit-flex-flow: row wrap;
+    justify-content: space-evenly;
+    align-items:center;
+    width: 100%;
+    cursor:pointer;
+}
+
+.japp .jtoolbar > div > div
+{
+    text-align:center;
+    flex:1;
+    margin:auto;
+}
+
+.japp .jtoolbar a
+{
+    text-decoration:none;
+    display:inline-block;
+}
+
+.japp .jtoolbar i
+{
+    color:#929292;
+}
+
+.japp .jtoolbar span
+{
+    font-size:0.7em;
+    display:block;
+    color:#929292;
+}
+
+.japp .jtoolbar .selected a,
+.japp .jtoolbar .selected i,
+.japp .jtoolbar .selected span
+{
+    color:var(--active-color);
+}
+
+
+/** Options **/
+
+.japp .options
+{
+    background-color:#fff;
+    border-top:1px solid #e6e6e8;
+    border-bottom:1px solid #e6e6e8;
+    padding-left:15px;
+    margin-top:10px;
+}
+
+.japp .options:empty
+{
+    display:none;
+}
+
+.japp .options .option
+{
+    padding-top:10px;
+    padding-bottom:10px;
+    text-transform:uppercase;
+    font-size:1em;
+    border-bottom:1px solid #e6e6e8;
+    overflow-x:hidden;
+
+    -webkit-box-sizing: border-box;
+    box-sizing: border-box;
+    vertical-align:middle;
+
+    display: -webkit-box;
+    display: -moz-box;
+    display: -ms-flexbox;
+    display: -webkit-flex;
+    display: flex;
+
+    -webkit-flex-flow: row nowrap;
+    justify-content: space-between;
+    align-items:center;
+    flex-wrap:nowrap;
+}
+
+.japp .options .option:last-child
+{
+    border-bottom:0px;
+}
+
+.japp .options label
+{
+    width:100%;
+}
+
+.japp .options input[type='text'], .japp .options textarea
+{
+    outline:none;
+    border:1px solid transparent;
+    font-size:1em;
+    margin:0px;
+    padding:0px;
+    width:100%;
+    box-sizing:border-box;
+}
+
+.japp .options textarea
+{
+    height:100px;
+}
+
+.japp .options input[type='checkbox'], .japp .options input[type='radio']
+{
+    visibility: hidden;
+}
+
+.japp .options input[type='checkbox'] ~ i, .japp .options input[type='radio'] ~ i
+{
+    background-image: url('');
+    background-repeat: no-repeat;
+    content:'';
+    visibility: hidden;
+    margin-right:10px;
+    line-height:24px;
+    width:24px;
+    height:24px;
+}
+
+.japp .options input[type='checkbox']:checked ~ i, .japp .options input[type='radio']:checked  ~ i
+{
+    visibility: visible;
+}
+
+.japp .options .jdropdown {
+    display: block;
+}
+
+.japp .options .jdropdown .jdropdown-header {
+    border: 0px;
+    border-bottom: 1px solid #e6e6e8;
+    padding: 15px;
+    text-transform: uppercase;
+}
+
+.japp .options .jdropdown:last-child .jdropdown-header {
+    border-bottom: 0px;
+}
+
+.japp .options .jdropdown-searchbar.jdropdown-focus .jdropdown-header {
+    border:0px;
+    padding: 5px;
+} 
+
+.japp .options .icon {
+    float:left;
+    margin-right:10px;
+    max-width:40px;
+    max-height:40px;
+    border-radius:20px;
+    color:#929292;
+}
+
+.japp .options .option .option-actions {
+    display:flex;
+    transform: translateX(100%);
+    width:0px;
+}
+
+.japp .options .option .option-actions > div {
+    padding-right:5px;
+}
+
+.japp .options .option .option-actions > div > i {
+    width:40px;
+    height:40px;
+    color:#fff;
+    font-size:24px;
+    line-height:40px;
+    text-align:center;
+    border-radius:30px;
+    background-color:red;
+}
+
+.japp .options .option .option-actions.small > div > i
+{
+    color:#000;
+    width:24px;
+    height:24px;
+    font-size:24px;
+    line-height:24px;
+    border-radius:24px;
+    background-color:transparent;
+}
+
+.japp .options .option-title {
+    display:block;
+    padding-top:10px;
+    padding-bottom:10px;
+    text-transform:uppercase;
+    font-size:1em;
+    vertical-align:center;
+}
+
+.japp .options .option-title::after {
+    content:'\e313';
+    font-family: 'material icons';
+    font-size:24px;
+    margin-right:10px;
+    float:right;
+    width:24px;
+    height:24px;
+    line-height:24px;
+}
+
+.japp .options .option-title.selected::after
+{
+    content:'\e316';
+}
+
+.japp .options .option-title.selected ~ div {
+    display:block;
+}
+
+.japp .options .option-group {
+    border-top:1px solid #e6e6e8;
+    background-color:#fff;
+    display:none;
+}
+
+.japp .options .option-link:after {
+    content:'\e315';
+    font-family: 'material icons';
+    color:var(--active-color);
+    font-size:24px;
+    margin-right:10px;
+    float:right;
+    width:24px;
+    height:24px;
+    line-height:24px;
+    display:block;
+}
+
+.japp .options .option-header {
+    padding-left:10px;
+    padding-right:10px;
+    flex-grow:10;
+    white-space: nowrap;
+    overflow: hidden;
+    text-overflow: ellipsis;
+}
+
+.japp .options .option .option-name {
+    font-size:1em;
+    text-overflow: ellipsis;
+    width: 100%;
+    overflow: hidden;
+}
+
+.japp .options .option .option-small {
+    font-size:0.7em;
+    color: #6d6d72;
+    text-overflow: ellipsis;
+    overflow: hidden;
+    white-space: nowrap;
+    display:block;
+}
+
+.japp .options .option .option-image img {
+    width:40px;
+    height:40px;
+    border-radius:20px;
+    display:block;
+}
+
+.japp .options .option .option-image-small img {
+    width:24px;
+    height:auto;
+    border-radius:50px;
+    display:block;
+}
+
+.japp .options .option .option-image .option-badge {
+    position: absolute;
+    background-color: red;
+    color: #fff;
+    width: 16px;
+    height: 16px;
+    line-height: 16px;
+    border-radius: 16px;
+    text-align: center;
+    font-size: 7px;
+    margin: 28px;
+}
+
+.japp .options .option .option-image-small .option-badge {
+    position: absolute;
+    background-color: red;
+    color: #fff;
+    width: 16px;
+    height: 16px;
+    line-height: 16px;
+    border-radius: 16px;
+    text-align: center;
+    font-size: 7px;
+    margin: 12px;
+}
+
+.japp .options .option .option-badge.solid {
+    background-color: red;
+    color: red;
+}
+
+.japp .options .option .option-badge.solid-green {
+    background-color: green;
+    color: green;
+}
+
+.japp .options .option .option-badge:empty {
+    display:none;
+}
+
+.japp .options .option .option-date {
+    float:right;
+    font-size:0.6em;
+    color:#888;
+    margin-right:15px;
+    white-space: nowrap;
+}
+
+.japp .options .option .option-right {
+    margin-right:15px;
+}
+
+.japp .options .option .option-badget {
+    background-color:red;
+    border-radius:12px;
+    width:24px;
+    height:24px;
+    line-height:24px;
+    font-size:0.7em;
+    color:#fff;
+    text-align:center;
+}
+
+.japp .options .option .option-badget:empty {
+    display:none;
+}
+
+/** Badge **/
+
+.japp .jbadge
+{
+    position:relative;
+    display:inline-block;
+    top:-12px;
+    left:-12px;
+}
+
+.japp .jbadge > div
+{
+    background-color:red;
+    color:#fff;
+    width:16px;
+    height:16px;
+    line-height:16px;
+    border-radius:16px;
+    text-align:center;
+    font-size:8px;
+    position:absolute;
+}
+
+.japp .jbadge > div:empty
+{
+    display:none;
+}
+
+/** Picker **/
+
+.japp .picker 
+{
+    background-color:#fff;
+    border-top:1px solid #e6e6e8;
+    border-bottom:1px solid #e6e6e8;
+    padding-left:15px;
+    margin-top:10px;
+    position:absolute;
+    bottom:0px;
+    width:100%;
+    z-index:700;
+}
+
+.japp .picker .picker-options
+{
+    -webkit-mask-image: linear-gradient(to bottom, rgba(0,0,0,1), rgba(0,0,0,.2)), linear-gradient(to top, rgba(0,0,0,1), rgba(0,0,0,.2));
+    -webkit-mask-size: 100% 50%;
+    -webkit-mask-repeat: no-repeat;
+    -webkit-mask-position: center bottom, top center;
+    overflow-y:scroll;
+    padding-top:100px;
+    padding-bottom:100px;
+}
+
+.japp .picker label
+{
+    display: block;
+    padding-top: 10px;
+    padding-bottom: 10px;
+}
+
+.japp .picker .picker-selected
+{
+    border-top:1px solid #e6e6e8;
+    border-bottom:1px solid #e6e6e8;
+}
+
+.japp .picker input
+{
+    display:none;
+}
+
+/** Topsearch **/
+
+.japp .top-search {
+    margin:0px;
+    padding:10px;
+    border-bottom:1px solid #ddd; 
+}
+
+.japp .top-search input {
+    background-color: #e6e6e6;
+    border: 0px;
+    border-radius: 4px;
+    outline: none;
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='gray'%3E%3Cpath d='M15.5 14h-.79l-.28-.27C15.41 12.59 16 11.11 16 9.5 16 5.91 13.09 3 9.5 3S3 5.91 3 9.5 5.91 16 9.5 16c1.61 0 3.09-.59 4.23-1.57l.27.28v.79l5 4.99L20.49 19l-4.99-5zm-6 0C7.01 14 5 11.99 5 9.5S7.01 5 9.5 5 14 7.01 14 9.5 11.99 14 9.5 14z'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E");
+    background-repeat: no-repeat;
+    background-position: 5px center;
+    padding-left: 32px;
+    width: 100%;
+}
+
+.japp .searchbar-container i {
+    position:absolute;
+    margin-top:5px;
+    margin-left:8px;
+    color:#bbb;
+}
+
+/** Range **/
+
+.japp .range
+{
+  -webkit-appearance: none;
+  margin: 18px 0;
+  width: 100%;
+}
+
+.japp .range:focus
+{
+  outline: none;
+}
+
+.japp .range::-webkit-slider-runnable-track
+{
+    width: 100%;
+    height: 4px;
+    cursor: pointer;
+    animation: 0.2s;
+    background: #ccc;
+    border-radius: 1.3px;
+}
+
+.japp .range::-webkit-slider-thumb
+{
+    -webkit-appearance:none;
+    appearance:none;
+    width:24px;
+    height:24px;
+    background:#fff;
+    cursor:pointer;
+    border-radius:24px;
+    margin-top:-10px;
+    border:0px;
+    box-shadow:0px 1px 3px 1px #bbb;
+}
+
+.japp .range::-moz-range-track
+{
+    width:100%;
+    cursor:pointer;
+    animation:0.2s;
+    background:#ccc;
+    border-radius:1.3px;
+}
+
+.japp .range::-moz-range-thumb
+{
+    width:24px;
+    height:24px;
+    background:#fff;
+    cursor:pointer;
+    border-radius:24px;
+    border:0px;
+    box-shadow:0px 1px 3px 1px #bbb;
+}
+
+.japp .range::-ms-track
+{
+    width: 100%;
+    cursor: pointer;
+    animation: 0.2s;
+    background: transparent;
+    border-color: transparent;
+    border-width: 16px 0;
+    color: transparent;
+    border-radius:1.3px;
+}
+
+.japp .range::-ms-fill-lower
+{
+    background:#ccc;
+    height:5px;
+}
+
+.japp .range::-ms-fill-upper
+{
+    background:#ccc;
+    height:5px;
+}
+
+.japp .range::-ms-thumb {
+    width:24px;
+    height:24px;
+    background:#fff;
+    cursor:pointer;
+    border-radius:24px;
+    border:0px;
+    box-shadow:0px 1px 3px 1px #bbb;
+}
+
+.japp .range:focus::-ms-fill-lower
+{
+  background: #ccc;
+}
+
+.japp .range:focus::-ms-fill-upper
+{
+  background: #ccc;
+}
+
+
+/** Actionsheet **/
+
+.jactionsheet
+{
+    position:fixed;
+    left: 0;
+    bottom: 0;
+    width: 100%;
+    min-height:100%;
+    z-index:10000;
+
+    background-color:rgba(0,0,0,0.5);
+    -webkit-transition-duration: .2s;
+    transition-duration: .2s;
+    display: flex;
+    -ms-flex-align: baseline;
+    -webkit-align-items: baseline;
+    -webkit-box-align: baseline;
+
+    align-items: baseline;
+}
+
+.jactionsheet .jactionsheet-content
+{
+    width:100%;
+    align-self: flex-end;
+}
+
+.jactionsheet .jactionsheet-title
+{
+    font-size:1em;
+    font-weight:bold;
+    padding:8px;
+}
+
+.jactionsheet .jactionsheet-message
+{
+    padding:4px;
+}
+
+.jactionsheet .jactionsheet-group
+{
+    box-sizing: border-box;
+    margin:10px;
+    background-color:#fff;
+    border-radius:5px;
+}
+
+.jactionsheet .jactionsheet-group > div
+{
+    font-size:1.4em;
+    font-weight:bold;
+    padding:4px;
+    border-top:1px solid #ccc;
+    text-align:center;
+    width:100%;
+}
+.jactionsheet .jactionsheet-group > div:first-child
+{
+    border-top:0px;
+}
+
+.jactionsheet .jactionsheet-group input
+{
+    border:0px;
+    margin:0px;
+    background-color:transparent;
+    color:var(--active-color);
+    width:100%;
+    outline:none;
+    font-size:1em;
+}
+
+.jactionsheet .jactionsheet-group input.jactionsheet-cancel
+{
+    color:red;
+}
+
+/** Placeholders **/
+
+.japp .options input::-webkit-input-placeholder
+{
+    text-transform:uppercase;
+}
+
+.japp .options input::-moz-placeholder
+{
+    text-transform:uppercase;
+}
+
+.japp .options input:-ms-input-placeholder
+{
+    text-transform:uppercase;
+}
+
+.japp .options input:-moz-placeholder
+{
+    text-transform:uppercase;
+}
+
+::placeholder
+{
+    color: #cccccc;
+    opacity: 1;
+}
+
+:-ms-input-placeholder
+{
+    color: #cccccc;
+}
+
+::-ms-input-placeholder
+{
+    color: #cccccc;
+}
+
+::-webkit-input-placeholder
+{
+    color: #cccccc;
+}
+
+
+.jmodal {
+    position:fixed;
+    top:50%;
+    left:50%;
+    width:60%;
+    height:60%;
+    -webkit-box-shadow: 0 2px 10px rgba(0,0,0,.2);
+    -moz-box-shadow: 0 2px 10px rgba(0,0,0,.2);
+    border:1px solid #ccc;
+    background-color:#fff;
+    transform: translate(-50%, -50%);
+    box-sizing: border-box;
+    padding-top:50px;
+    z-index:9002;
+    border-radius:5px;
+}
+
+.jmodal:before {
+    position:absolute;
+    top:0;
+    left:0;
+    width:100%;
+    content:attr(title);
+    padding:15px;
+    box-sizing: border-box;
+    background: #e3e3e3;
+    filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ededed', endColorstr='#e3e3e3');
+    background: -webkit-gradient(linear, left top, left bottom, from(#ededed), to(#e3e3e3));
+    background: -moz-linear-gradient(top,  #ededed,  #e3e3e3);
+    font-size:1.2em;
+}
+
+.jmodal_content {
+    padding:20px;
+    overflow-y:auto;
+    max-height:100%;
+    box-sizing: border-box;
+}
+.jmodal.no-title {
+    padding-top:0px;
+}
+
+.jmodal.no-title:before {
+    display:none;
+}
+
+.jmodal:after {
+    content:'';
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E");
+    position:absolute;
+    top:0;
+    right:0;
+    margin:14px;
+    font-size:24px;
+    width:24px;
+    height:24px;
+    cursor:pointer;
+    text-shadow: 0px 0px 5px #fff;
+}
+
+.jmodal_fullscreen {
+    width: 100% !important;
+    height: 100% !important;
+    top: 0px;
+    left: 0px;
+    transform: none;
+    border-radius: 0px;
+}
+
+
+.jmodal_backdrop {
+    position: fixed;
+    top: 0px;
+    left: 0px;
+    min-width: 100%;
+    min-height: 100%;
+    background-color: rgba(0,0,0,0.5);
+    border: 0px;
+    padding: 0px;
+    z-index: 8000;
+    display: none;
+
+  -webkit-touch-callout: none; /* iOS Safari */
+    -webkit-user-select: none; /* Safari */
+     -khtml-user-select: none; /* Konqueror HTML */
+       -moz-user-select: none; /* Firefox */
+        -ms-user-select: none; /* Internet Explorer/Edge */
+            user-select: none; /* Non-prefixed version, currently
+                                  supported by Chrome and Opera */
+}
+
+
+.jnotification {
+    position: fixed;
+    z-index: 10000;
+    -webkit-box-sizing: border-box;
+    box-sizing: border-box;
+    padding: 10px;
+}
+
+.jnotification-container {
+    -webkit-box-shadow: 0px 2px 15px -5px rgba(0, 0, 0, 0.7);
+    box-shadow: 0px 2px 15px -5px rgba(0, 0, 0, 0.7);
+    padding: 12px;
+    border-radius: 8px;
+}
+
+.jnotification-close {
+    content: '';
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='20' height='20' viewBox='0 0 24 24' fill='gray'%3E%3Cpath d='M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E");
+    font-size: 20px;
+    width: 20px;
+    height: 20px;
+    cursor: pointer;
+}
+
+.jnotification-title {
+    font-weight: bold;
+}
+
+.jnotification-header {
+    display: flex;
+    padding-bottom: 5px;
+}
+
+.jnotification-header:empty {
+    display: none;
+}
+
+.jnotification-image {
+    margin-right: 5px;
+}
+
+.jnotification-image:empty {
+    display: none;
+}
+
+.jnotification-image img {
+    width: 24px;
+}
+
+.jnotification-name {
+    text-transform: uppercase;
+    font-size: 0.9em;
+    flex: 1;
+    letter-spacing: 0.1em;
+}
+
+@media (min-width: 320px) and (max-width: 800px) {
+    .jnotification {
+        top: 0px;
+        width: 100%;
+    }
+    .jnotification-container {
+        background: rgba(255,255,255,0.95);
+        border: 1px solid #eee;
+    }
+}
+
+@media (min-width: 801px) {
+    .jnotification {
+        bottom: 0px;
+    }
+    .jnotification-container {
+        background-color: #000;
+        background: rgba(92,92,92,1);
+        background: linear-gradient(0deg, rgba(92,92,92,1) 0%, rgba(77,77,77,1) 100%);
+        color: #fff;
+        width: 320px;
+        margin: 30px;
+        padding: 20px;
+    }
+}
+
+.jnotification-header {
+    display: -webkit-box;
+    display: -webkit-flex;
+    display: -ms-flexbox;
+    display: flex;
+    -webkit-box-pack: start;
+    -webkit-justify-content: flex-start;
+    -ms-flex-pack: start;
+    justify-content: flex-start;
+    -webkit-box-align: center;
+    -webkit-align-items: center;
+    -ms-flex-align: center;
+    align-items: center;
+}
+
+.jrating {
+    display:flex;
+}
+.jrating > div {
+    width:24px;
+    height:24px;
+    line-height:24px;
+    background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M22 9.24l-7.19-.62L12 2 9.19 8.63 2 9.24l5.46 4.73L5.82 21 12 17.27 18.18 21l-1.63-7.03L22 9.24zM12 15.4l-3.76 2.27 1-4.28-3.32-2.88 4.38-.38L12 6.1l1.71 4.04 4.38.38-3.32 2.88 1 4.28L12 15.4z' fill='gray'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E");
+}
+
+.jrating .jrating-over {
+    background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='black'%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3Cpath d='M12 17.27L18.18 21l-1.64-7.03L22 9.24l-7.19-.61L12 2 9.19 8.63 2 9.24l5.46 4.73L5.82 21z'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E");
+    opacity: 0.7;
+}
+
+.jrating .jrating-selected {
+    background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='red'%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3Cpath d='M12 17.27L18.18 21l-1.64-7.03L22 9.24l-7.19-.61L12 2 9.19 8.63 2 9.24l5.46 4.73L5.82 21z'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E");
+}
+
+
+/**
+ * (c) Image slider
+ * https://github.com/paulhodel/jtools
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Image Slider
+ */
+
+.jslider {
+    margin-top:10px;
+    margin-bottom:10px;
+}
+
+.jslider-container img {
+    width:60px;
+    margin-right:5px;
+    margin-bottom:5px;
+}
+
+.jslider-attach {
+    cursor:pointer;
+}
+
+.jslider-left::before {
+    position: fixed;
+    left: 10px;
+    content:'arrow_back_ios';
+    color: #fff;
+    width: 30px;
+    height: 30px;
+    font-family: 'Material Icons';
+    font-size: 30px;
+    text-shadow: 0px 0px 0px #000;
+    text-align: center;
+}
+
+.jslider-right::after {
+    position: fixed;
+    right: 5px;
+    content: 'arrow_forward_ios';
+    color: #fff;
+    width: 30px;
+    height: 30px;
+    font-family: 'Material Icons';
+    font-size: 30px;
+    text-shadow: 0px 0px 0px #000;
+    text-align: center;
+}
+
+.jslider-close {
+    width:24px;
+    height:24px;
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24' fill='white'%3E%3Cpath d='M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E");
+    position:fixed;
+    top:15px;
+    right:15px;
+    display:none;
+    cursor:pointer;
+    z-index:3000;
+}
+
+.jslider-preview {
+    position:fixed;
+    left:0;
+    top:0;
+    width: 100%;
+    min-height:100%;
+    z-index:2000;
+    margin:0px;
+    box-sizing:border-box;
+
+    background-color:rgba(0,0,0,0.9);
+    -webkit-transition-duration: .4s;
+    transition-duration: .4s;
+    display: flex;
+    -ms-flex-align: center;
+    -webkit-align-items: center;
+    -webkit-box-align: center;
+
+    align-items: center;
+}
+
+.jslider-preview img {
+    min-width:70%;
+    max-width:100%;
+    height:auto;
+    box-sizing: border-box;
+    margin:0 auto;
+    vertical-align:middle;
+    display:none;
+}
+
+
+.jswitch {
+  display: inline-block;
+  cursor: pointer;
+  -webkit-tap-highlight-color: transparent;
+}
+
+.jswitch i {
+  position: relative;
+  display: inline-block;
+ margin-right: .5rem;
+  width: 46px;
+  height: 26px;
+  background-color: #e6e6e6;
+  border-radius: 23px;
+  vertical-align: text-bottom;
+  transition: all 0.3s linear;
+}
+
+.jswitch i::before {
+  content: "";
+  position: absolute;
+  left: 0;
+  width: 42px;
+  height: 22px;
+  background-color: #fff;
+  border-radius: 11px;
+  transform: translate3d(2px, 2px, 0) scale3d(1, 1, 1);
+  transition: all 0.25s linear;
+}
+
+.jswitch i::after {
+  content: "";
+  position: absolute;
+  left: 0;
+  width: 22px;
+  height: 22px;
+  background-color: #fff;
+  border-radius: 11px;
+  box-shadow: 0 2px 2px rgba(0, 0, 0, 0.24);
+  transform: translate3d(2px, 2px, 0);
+  transition: all 0.2s ease-in-out;
+}
+
+.jswitch:active i::after {
+  width: 28px;
+  transform: translate3d(2px, 2px, 0);
+}
+
+.jswitch:active input:checked + i::after { transform: translate3d(16px, 2px, 0); }
+
+.jswitch input { display: none; }
+
+.jswitch input:checked + i { background-color: #4BD763; }
+
+.jswitch input:checked + i::before { transform: translate3d(18px, 2px, 0) scale3d(0, 0, 0); }
+
+.jswitch input:checked + i::after { transform: translate3d(22px, 2px, 0); }
+
+.jtabs {
+    max-width: calc(100vw - 24px);
+    position: relative;
+}
+.jtabs > .jtabs-headers {
+    display: flex;
+    overflow-x: auto;
+}
+
+.jtabs > .jtabs-headers > div {
+    padding: 6px;
+    padding-left: 20px;
+    padding-right: 20px;
+    margin-left: 1px;
+    margin-right: 1px;
+    margin-bottom: 1px;
+    background-color: #f1f1f1;
+    cursor: pointer;
+}
+
+.jtabs > .jtabs-headers > div.jtabs-selected {
+    background-color: #e8e8e8;
+    color: #000;
+}
+
+.jtabs > .jtabs-content > div {
+    display: none;
+}
+
+.jtabs > .jtabs-content > div.jtabs-selected {
+    display: block;
+}
+
+.jtabs > .jtabs-border {
+    position: absolute;
+    height: 2px;
+    background-color: #888;
+    transform-origin: left;
+    transition: all .2s cubic-bezier(0.4,0,0.2,1);
+    transition-property: color,left,transform;
+}
+
+.jtabs .jtabs-add {
+    background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' height='24' viewBox='0 0 24 24' width='24'%3E%3Cpath d='M19 3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm-2 10h-4v4h-2v-4H7v-2h4V7h2v4h4v2z' fill='%23bbbbbb'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E");
+    cursor: pointer;
+    background-position: center;
+    background-repeat: no-repeat;
+    width: 24px;
+    height: 24px;
+    line-height: 24px;
+    margin: 3px;
+    margin-left: 10px;
+}
+
+.jtags {
+    display: flex;
+    flex-wrap: wrap;
+    -ms-flex-direction: row;
+    -webkit-flex-direction: row;
+    flex-direction: row;
+    -ms-flex-pack: flex-start;
+    -webkit-justify-content: space-between;
+    justify-content: flex-start;
+    padding: 2px;
+    border: 1px solid #ccc;
+} 
+
+.jtags > div {
+    padding: 3px;
+    padding-left: 10px;
+    padding-right: 22px;
+    position: relative;
+    border-radius: 1px;
+    margin: 2px;
+    display: block;
+    outline: none;
+}
+
+.jtags > div::after {
+    content: 'x';
+    position: absolute;
+    top: 4px;
+    right: 4px;
+    width: 12px;
+    height: 12px;
+    cursor: pointer;
+    font-size: 11px;
+    display: none;
+}
+
+.jtags_label {
+    background-color: #eeeeee !important;
+}
+
+.jtags_label::after {
+    display: inline-block !important;
+}
+
+.jtags_error::after {
+    color: #fff  !important;
+}
+
+.jtags_error {
+    background-color: #d93025 !important;
+    color: #fff;
+}
+
+
+.jtags_search {
+    position: absolute;
+    display: none;
+    box-shadow: 0 1px 2px 0 rgba(60,64,67,0.302), 0 2px 6px 2px rgba(60,64,67,0.149);
+    border: none;
+    -webkit-border-radius: 4px;
+    border-radius: 4px;
+    width: 280px;
+    padding: 8px 0;
+
+    -webkit-box-shadow: 0 2px 4px rgba(0,0,0,0.2);
+    box-shadow: 0 2px 4px rgba(0,0,0,0.2);
+    -webkit-transition: opacity .218s;
+    transition: opacity .218s;
+    background: #fff;
+    border: 1px solid rgba(0,0,0,.2);
+    cursor: pointer;
+    margin: 0;
+    min-width: 300px;
+    outline: none;
+    width: auto;
+    user-select: none;
+}
+
+.jtags_search > div {
+    color: #333;
+    cursor: pointer;
+    display: -webkit-box;
+    display: -webkit-flex;
+    display: flex;
+    padding: 5px 10px;
+    user-select: none;
+    -webkit-align-items: center;
+    align-items: center;
+}
+
+.jtags_search > div:hover {
+    background-color: #e8eaed;
+}
+
+.jtags_search > div > img {
+    width: 32px;
+    height: 32px;
+    user-select: none;
+    border-radius: 16px;
+    margin-right: 2px;
+}
+
+.jtags_search > div > div {
+    overflow: hidden;
+    text-overflow: ellipsis;
+    margin-left: 2px;
+    max-width: 300px;
+    white-space: nowrap;
+    user-select: none;
+}
+
+.jtags_search .selected {
+    background-color: #e8eaed;
+}
+
+/**
+ * (c) jTools Template renderer
+ * https://github.com/paulhodel/jtools
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Template renderer
+ */
+
+.jtemplate-pagination {
+    display:flex;
+    cursor:pointer;
+    margin-top:10px;
+    margin-bottom:10px;
+}
+
+.jtemplate-pagination div {
+    text-align:center;
+    font-size:0.8em;
+    width:30px;
+    height:30px;
+    line-height:30px;
+    -webkit-touch-callout: none; /* iOS Safari */
+    -webkit-user-select: none; /* Safari */
+    khtml-user-select: none; /* Konqueror HTML */
+    -moz-user-select: none; /* Firefox */
+    -ms-user-select: none; /* Internet Explorer/Edge */
+    user-select: none;
+}
+
+.jtemplate-pagination div:hover {
+    background-color:#eee;
+}
+
+.jtemplate-results {
+    margin-top:15px;
+    margin-bottom:15px;
+}
+
+.app .jtemplate-empty {
+    text-align:center;
+    padding-top:10px;
+    padding-bottom:10px;
+    margin-right:10px;
+}
+
+
+
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.css.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.css.xml
new file mode 100644
index 0000000000000000000000000000000000000000..c2a7e41dab0e20f582da6b3373c2b7adc7971851
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.css.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>jsuites.css</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>text/css</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.js.js b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.js.js
new file mode 100644
index 0000000000000000000000000000000000000000..b01ba28ba8bc929b621886eaa2f77ee73cf5fef1
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.js.js
@@ -0,0 +1,8949 @@
+/**
+ * (c) jSuites Javascript Web Components (v2.7)
+ *
+ * Author: Paul Hodel <paul.hodel@gmail.com>
+ * Website: https://bossanova.uk/jsuites/
+ * Description: Create amazing web based applications.
+ *
+ * MIT License
+ *
+ */
+;(function (global, factory) {
+    typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
+    typeof define === 'function' && define.amd ? define(factory) :
+    global.jSuites = factory();
+}(this, (function () {
+
+    'use strict';
+
+var jSuites = function(options) {
+    var obj = {}
+
+    obj.init = function() {
+        // Find root element
+        var app = document.querySelector('.japp');
+
+        // Root element
+        if (app) {
+            obj.el = app;
+        } else {
+            obj.el = document.body;
+        }
+
+        // Loading modules
+        var modules = document.querySelectorAll('[data-autoload]');
+        for (var i = 0; i < modules.length; i++) {
+            var m = modules[i].getAttribute('data-autoload');
+            if (typeof(window[m]) == 'function') {
+                window[m](modules[i]);
+            }
+        }
+    }
+
+    obj.guid = function() {
+        var guid = '';
+        for (var i = 0; i < 32; i++) {
+            guid += Math.floor(Math.random()*0xF).toString(0xF);
+        }
+        return guid;
+    }
+
+    obj.getWindowWidth = function() {
+        var w = window,
+        d = document,
+        e = d.documentElement,
+        g = d.getElementsByTagName('body')[0],
+        x = w.innerWidth || e.clientWidth || g.clientWidth;
+        return x;
+    }
+
+    obj.getWindowHeight = function() {
+        var w = window,
+        d = document,
+        e = d.documentElement,
+        g = d.getElementsByTagName('body')[0],
+        y = w.innerHeight|| e.clientHeight|| g.clientHeight;
+        return  y;
+    }
+
+    obj.getPosition = function(e) {
+        if (e.changedTouches && e.changedTouches[0]) {
+            var x = e.changedTouches[0].pageX;
+            var y = e.changedTouches[0].pageY;
+        } else {
+            var x = (window.Event) ? e.pageX : e.clientX + (document.documentElement.scrollLeft ? document.documentElement.scrollLeft : document.body.scrollLeft);
+            var y = (window.Event) ? e.pageY : e.clientY + (document.documentElement.scrollTop ? document.documentElement.scrollTop : document.body.scrollTop);
+        }
+
+        return [ x, y ];
+    }
+
+    obj.click = function(el) {
+        if (el.click) {
+            el.click();
+        } else {
+            var evt = new MouseEvent('click', {
+                bubbles: true,
+                cancelable: true,
+                view: window
+            });
+            el.dispatchEvent(evt);
+        }
+    }
+
+    obj.getElement = function(element, className) {
+        var foundElement = false;
+
+        function path (element) {
+            if (element.className) {
+                if (element.classList.contains(className)) {
+                    foundElement = element;
+                }
+            }
+
+            if (element.parentNode) {
+                path(element.parentNode);
+            }
+        }
+
+        path(element);
+
+        return foundElement;
+    }
+
+    obj.getLinkElement = function(element) {
+        var targetElement = false;
+
+        function path (element) {
+            if ((element.tagName == 'A' || element.tagName == 'DIV') && element.getAttribute('data-href')) {
+                targetElement = element;
+            }
+
+            if (element.parentNode) {
+                path(element.parentNode);
+            }
+        }
+
+        path(element);
+
+        return targetElement;
+    }
+
+    obj.getFormElements = function(formObject) {
+        var ret = {};
+
+        if (formObject) {
+            var elements = formObject.querySelectorAll("input, select, textarea");
+        } else {
+            var elements = document.querySelectorAll("input, select, textarea");
+        }
+
+        for (var i = 0; i < elements.length; i++) {
+            var element = elements[i];
+            var name = element.name;
+            var value = element.value;
+
+            if (name) {
+                ret[name] = value;
+            }
+        }
+
+        return ret;
+    }
+
+    obj.exists = function(url, __callback) {
+        var http = new XMLHttpRequest();
+        http.open('HEAD', url, false);
+        http.send();
+        if (http.status) {
+            __callback(http.status);
+        }
+    }
+
+    obj.getFiles = function(element) {
+        if (! element) {
+            console.error('No element defined in the arguments of your method');
+        }
+
+        // Get attachments
+        var files = element.querySelectorAll('.jfile');
+
+        if (files.length > 0) {
+            var data = [];
+            for (var i = 0; i < files.length; i++) {
+                var file = {};
+
+                var src = files[i].getAttribute('src');
+
+                if (files[i].classList.contains('jremove')) {
+                    file.remove = 1;
+                } else {
+                    if (src.substr(0,4) == 'data') {
+                        file.content = src.substr(src.indexOf(',') + 1);
+                        file.extension = files[i].getAttribute('data-extension');
+                    } else {
+                        file.file = src;
+                        file.extension = files[i].getAttribute('data-extension');
+                        if (! file.extension) {
+                            file.extension =  src.substr(src.lastIndexOf('.') + 1);
+                        }
+                        if (jSuites.files[file.file]) {
+                            file.content = jSuites.files[file.file];
+                        }
+                    }
+
+                    // Optional file information
+                    if (files[i].getAttribute('data-name')) {
+                        file.name = files[i].getAttribute('data-name');
+                    }
+
+                    if (files[i].getAttribute('data-file')) {
+                        file.file = files[i].getAttribute('data-file');
+                    }
+
+                    if (files[i].getAttribute('data-size')) {
+                        file.size = files[i].getAttribute('data-size');
+                    }
+
+                    if (files[i].getAttribute('data-date')) {
+                        file.date = files[i].getAttribute('data-date');
+                    }
+
+                    if (files[i].getAttribute('data-cover')) {
+                        file.cover = files[i].getAttribute('data-cover');
+                    }
+                }
+
+                // TODO SMALL thumbs?
+
+                data[i] = file;
+            }
+
+            return data;
+        }
+    }
+
+    obj.ajax = function(options) {
+        if (! options.data) {
+            options.data = {};
+        }
+
+        if (options.type) {
+            options.method = options.type;
+        }
+
+        if (options.data) {
+            var data = [];
+            var keys = Object.keys(options.data);
+
+            if (keys.length) {
+                for (var i = 0; i < keys.length; i++) {
+                    if (typeof(options.data[keys[i]]) == 'object') {
+                        var o = options.data[keys[i]];
+                        for (var j = 0; j < o.length; j++) {
+                            if (typeof(o[j]) == 'string') {
+                                data.push(keys[i] + '[' + j + ']=' + encodeURIComponent(o[j]));
+                            } else {
+                                var prop = Object.keys(o[j]);
+                                for (var z = 0; z < prop.length; z++) {
+                                    data.push(keys[i] + '[' + j + '][' + prop[z] + ']=' + encodeURIComponent(o[j][prop[z]]));
+                                }
+                            }
+                        }
+                    } else {
+                        data.push(keys[i] + '=' + encodeURIComponent(options.data[keys[i]]));
+                    }
+                }
+            }
+
+            if (options.method == 'GET' && data.length > 0) {
+                if (options.url.indexOf('?') < 0) {
+                    options.url += '?';
+                }
+                options.url += data.join('&');
+            }
+        }
+
+        var httpRequest = new XMLHttpRequest();
+        httpRequest.open(options.method, options.url, true);
+        httpRequest.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
+
+        if (options.method == 'POST') {
+            httpRequest.setRequestHeader('Accept', 'application/json');
+            httpRequest.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
+        } else {
+            if (options.dataType == 'json') {
+                httpRequest.setRequestHeader('Content-Type', 'text/json');
+            }
+        }
+
+        // No cache
+        if (options.cache != true) {
+            httpRequest.setRequestHeader('pragma', 'no-cache');
+            httpRequest.setRequestHeader('cache-control', 'no-cache');
+        }
+
+        // Authentication
+        if (options.withCredentials == true) {
+            httpRequest.withCredentials = true
+        }
+
+        // Before send
+        if (typeof(options.beforeSend) == 'function') {
+            options.beforeSend(httpRequest);
+        }
+
+        httpRequest.onload = function() {
+            if (httpRequest.status === 200) {
+                if (options.dataType == 'json') {
+                    try {
+                        var result = JSON.parse(httpRequest.responseText);
+
+                        if (options.success && typeof(options.success) == 'function') {
+                            options.success(result);
+                        }
+                    } catch(err) {
+                        if (options.error && typeof(options.error) == 'function') {
+                            options.error(result);
+                        }
+                    }
+                } else {
+                    var result = httpRequest.responseText;
+
+                    if (options.success && typeof(options.success) == 'function') {
+                        options.success(result);
+                    }
+                }
+            } else {
+                if (options.error && typeof(options.error) == 'function') {
+                    options.error(httpRequest.responseText);
+                }
+            }
+
+            // Global complete method
+            if (obj.ajax.requests && obj.ajax.requests.length) {
+                // Get index of this request in the container
+                var index = obj.ajax.requests.indexOf(httpRequest)
+                // Remove from the ajax requests container
+                obj.ajax.requests.splice(index, 1);
+                // Last one?
+                if (! obj.ajax.requests.length) {
+                    if (options.complete && typeof(options.complete) == 'function') {
+                        options.complete(result);
+                    }
+                }
+            }
+        }
+
+        if (data) {
+            httpRequest.send(data.join('&'));
+        } else {
+            httpRequest.send();
+        }
+
+        obj.ajax.requests.push(httpRequest);
+
+        return httpRequest;
+    }
+
+    obj.ajax.requests = [];
+
+    obj.slideLeft = function(element, direction, done) {
+        if (direction == true) {
+            element.classList.add('slide-left-in');
+            setTimeout(function() {
+                element.classList.remove('slide-left-in');
+                if (typeof(done) == 'function') {
+                    done();
+                }
+            }, 400);
+        } else {
+            element.classList.add('slide-left-out');
+            setTimeout(function() {
+                element.classList.remove('slide-left-out');
+                if (typeof(done) == 'function') {
+                    done();
+                }
+            }, 400);
+        }
+    }
+
+    obj.slideRight = function(element, direction, done) {
+        if (direction == true) {
+            element.classList.add('slide-right-in');
+            setTimeout(function() {
+                element.classList.remove('slide-right-in');
+                if (typeof(done) == 'function') {
+                    done();
+                }
+            }, 400);
+        } else {
+            element.classList.add('slide-right-out');
+            setTimeout(function() {
+                element.classList.remove('slide-right-out');
+                if (typeof(done) == 'function') {
+                    done();
+                }
+            }, 400);
+        }
+    }
+
+    obj.slideTop = function(element, direction, done) {
+        if (direction == true) {
+            element.classList.add('slide-top-in');
+            setTimeout(function() {
+                element.classList.remove('slide-top-in');
+                if (typeof(done) == 'function') {
+                    done();
+                }
+            }, 400);
+        } else {
+            element.classList.add('slide-top-out');
+            setTimeout(function() {
+                element.classList.remove('slide-top-out');
+                if (typeof(done) == 'function') {
+                    done();
+                }
+            }, 400);
+        }
+    }
+
+    obj.slideBottom = function(element, direction, done) {
+        if (direction == true) {
+            element.classList.add('slide-bottom-in');
+            setTimeout(function() {
+                element.classList.remove('slide-bottom-in');
+                if (typeof(done) == 'function') {
+                    done();
+                }
+            }, 400);
+        } else {
+            element.classList.add('slide-bottom-out');
+            setTimeout(function() {
+                element.classList.remove('slide-bottom-out');
+                if (typeof(done) == 'function') {
+                    done();
+                }
+            }, 100);
+        }
+    }
+
+    obj.fadeIn = function(element, done) {
+        element.classList.add('fade-in');
+        setTimeout(function() {
+            element.classList.remove('fade-in');
+            if (typeof(done) == 'function') {
+                done();
+            }
+        }, 2000);
+    }
+
+    obj.fadeOut = function(element, done) {
+        element.classList.add('fade-out');
+        setTimeout(function() {
+            element.classList.remove('fade-out');
+            if (typeof(done) == 'function') {
+                done();
+            }
+        }, 1000);
+    }
+
+    obj.keyDownControls = function(e) {
+        if (e.which == 27) {
+            var nodes = document.querySelectorAll('.jslider');
+            if (nodes.length > 0) {
+                for (var i = 0; i < nodes.length; i++) {
+                    nodes[i].slider.close();
+                }
+            }
+
+            if (document.querySelector('.jdialog')) {
+                jSuites.dialog.close();
+            }
+        } else if (e.which == 13) {
+            if (document.querySelector('.jdialog')) {
+                if (typeof(jSuites.dialog.options.onconfirm) == 'function') {
+                    jSuites.dialog.options.onconfirm();
+                }
+                jSuites.dialog.close();
+            }
+        }
+
+        // Verify mask
+        if (jSuites.mask) {
+            jSuites.mask.apply(e);
+        }
+    }
+
+    var actionUpControl = function(e) {
+        var element = null;
+        if (element = jSuites.getLinkElement(e.target)) {
+            var link = element.getAttribute('data-href');
+            if (link == '#back') {
+                window.history.back();
+            } else if (link == '#panel') {
+                jSuites.panel();
+            } else {
+                jSuites.pages(link);
+            }
+        }
+    }
+
+    var controlSwipeLeft = function(e) {
+        var element = jSuites.getElement(e.target, 'option');
+
+        if (element && element.querySelector('.option-actions')) {
+            element.scrollTo({
+                left: 100,
+                behavior: 'smooth'
+            });
+        } else {
+            var element = jSuites.getElement(e.target, 'jcalendar');
+            if (element && jSuites.calendar.current) {
+                jSuites.calendar.current.prev();
+            } else {
+                if (jSuites.panel) {
+                    var element = jSuites.panel.get();
+                    if (element) {
+                        if (element.style.display != 'none') {
+                            jSuites.panel.close();
+                        }
+                    }
+                }
+            }
+        }
+    }
+
+    var controlSwipeRight = function(e) {
+        var element = jSuites.getElement(e.target, 'option');
+        if (element && element.querySelector('.option-actions')) {
+            element.scrollTo({
+                left: 0,
+                behavior: 'smooth'
+            });
+        } else {
+            var element = jSuites.getElement(e.target, 'jcalendar');
+            if (element && jSuites.calendar.current) {
+                jSuites.calendar.current.next();
+            } else {
+                if (jSuites.panel) {
+                    var element = jSuites.panel.get();
+                    if (element) {
+                        if (element.style.display == 'none') {
+                            jSuites.panel();
+                        }
+                    }
+                }
+            }
+        }
+    }
+
+    var actionOverControl = function(e) {
+        // Tooltip
+        if (jSuites.tooltip) {
+            jSuites.tooltip(e);
+        }
+    }
+
+    var actionOutControl = function(e) {
+        // Tooltip
+        if (jSuites.tooltip) {
+            jSuites.tooltip.hide();
+        }
+    }
+
+    // Create page container
+    document.addEventListener('swipeleft', controlSwipeLeft);
+    document.addEventListener('swiperight', controlSwipeRight);
+    document.addEventListener('keydown', obj.keyDownControls);
+
+    if ('ontouchend' in document.documentElement === true) {
+        document.addEventListener('touchend', actionUpControl);
+    } else {
+        document.addEventListener('mouseup', actionUpControl);
+    }
+
+    // Onmouseover
+    document.addEventListener('mouseover', actionOverControl);
+    document.addEventListener('mouseout', actionOutControl);
+    document.addEventListener('DOMContentLoaded', function() {
+        obj.init();
+    });
+
+    // Pop state control
+    window.onpopstate = function(e) {
+        if (e.state && e.state.route) {
+            if (jSuites.pages.get(e.state.route)) {
+                jSuites.pages(e.state.route, { ignoreHistory:true });
+            }
+        }
+    }
+
+    return obj;
+}();
+
+jSuites.files = [];
+
+jSuites.calendar = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // Global container
+    if (! jSuites.calendar.current) {
+        jSuites.calendar.current = null;
+    }
+
+    // Default configuration
+    var defaults = {
+        // Data
+        data: null,
+        // Inline or not
+        type: null,
+        // Restrictions
+        validRange: null,
+        // Starting weekday - 0 for sunday, 6 for saturday
+        startingDay: null, 
+        // Date format
+        format: 'DD/MM/YYYY',
+        // Allow keyboard date entry
+        readonly: true,
+        // Today is default
+        today: false,
+        // Show timepicker
+        time: false,
+        // Show the reset button
+        resetButton: true,
+        // Placeholder
+        placeholder: '',
+        // Translations can be done here
+        months: ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'],
+        weekdays: ['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday'],
+        weekdays_short: ['S', 'M', 'T', 'W', 'T', 'F', 'S'],
+        // Value
+        value: null,
+        // Events
+        onclose: null,
+        onchange: null,
+        // Fullscreen (this is automatic set for screensize < 800)
+        fullscreen: false,
+        // Internal mode controller
+        mode: null,
+        position: null,
+        // Create the calendar closed as default
+        opened: false,
+    };
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Value
+    if (! obj.options.value) {
+        if (el.tagName == 'INPUT' && el.value) {
+            obj.options.value = el.value;
+        }
+    }
+
+    // Make sure use upper case in the format
+    obj.options.format = obj.options.format.toUpperCase();
+
+    if (obj.options.value) {
+        var date = obj.options.value.split(' ');
+        var time = date[1];
+        var date = date[0].split('-');
+        var y = parseInt(date[0]);
+        var m = parseInt(date[1]);
+        var d = parseInt(date[2]);
+
+        if (time) {
+            var time = time.split(':');
+            var h = parseInt(time[0]);
+            var i = parseInt(time[1]);
+        } else {
+            var h = 0;
+            var i = 0;
+        }
+    } else {
+        var date = new Date();
+        var y = date.getFullYear();
+        var m = date.getMonth() + 1;
+        var d = date.getDate();
+        var h = date.getHours();
+        var i = date.getMinutes();
+    }
+
+    // Current value
+    obj.date = [ y, m, d, h, i, 0 ];
+
+    // Two digits
+    var two = function(value) {
+        value = '' + value;
+        if (value.length == 1) {
+            value = '0' + value;
+        }
+        return value;
+    }
+
+    // Calendar elements
+    var calendarReset = document.createElement('div');
+    calendarReset.className = 'jcalendar-reset';
+    calendarReset.innerHTML = 'Reset';
+
+    var calendarConfirm = document.createElement('div');
+    calendarConfirm.className = 'jcalendar-confirm';
+    calendarConfirm.innerHTML = 'Done';
+
+    var calendarControls = document.createElement('div');
+    calendarControls.className = 'jcalendar-controls'
+    if (obj.options.resetButton) {
+        calendarControls.appendChild(calendarReset);
+    }
+    calendarControls.appendChild(calendarConfirm);
+
+    var calendarContainer = document.createElement('div');
+    calendarContainer.className = 'jcalendar-container';
+
+    var calendarContent = document.createElement('div');
+    calendarContent.className = 'jcalendar-content';
+    calendarContent.appendChild(calendarControls);
+    calendarContainer.appendChild(calendarContent);
+
+    // Table container
+    var calendarTableContainer = document.createElement('div');
+    calendarTableContainer.className = 'jcalendar-table';
+    calendarContent.appendChild(calendarTableContainer);
+
+    // Main element
+    if (el.tagName == 'INPUT') {
+        var calendar = document.createElement('div');
+    } else {
+        var calendar = el;
+    }
+    calendar.className = 'jcalendar';
+    calendar.appendChild(calendarContainer);
+
+    // Previous button
+    var calendarHeaderPrev = document.createElement('td');
+    calendarHeaderPrev.setAttribute('colspan', '2');
+    calendarHeaderPrev.className = 'jcalendar-prev';
+
+    // Header with year and month
+    var calendarLabelYear = document.createElement('span');
+    calendarLabelYear.className = 'jcalendar-year';
+
+    var calendarLabelMonth = document.createElement('span');
+    calendarLabelMonth.className = 'jcalendar-month';
+
+    var calendarHeaderTitle = document.createElement('td');
+    calendarHeaderTitle.className = 'jcalendar-header';
+    calendarHeaderTitle.setAttribute('colspan', '3');
+    calendarHeaderTitle.appendChild(calendarLabelMonth);
+    calendarHeaderTitle.appendChild(calendarLabelYear);
+
+    var calendarHeaderNext = document.createElement('td');
+    calendarHeaderNext.setAttribute('colspan', '2');
+    calendarHeaderNext.className = 'jcalendar-next';
+
+    var calendarHeaderRow = document.createElement('tr');
+    calendarHeaderRow.appendChild(calendarHeaderPrev);
+    calendarHeaderRow.appendChild(calendarHeaderTitle);
+    calendarHeaderRow.appendChild(calendarHeaderNext);
+
+    var calendarHeader = document.createElement('thead');
+    calendarHeader.appendChild(calendarHeaderRow);
+
+    var calendarBody = document.createElement('tbody');
+    var calendarFooter = document.createElement('tfoot');
+
+    // Calendar table
+    var calendarTable = document.createElement('table');
+    calendarTable.setAttribute('cellpadding', '0');
+    calendarTable.setAttribute('cellspacing', '0');
+    calendarTable.appendChild(calendarHeader);
+    calendarTable.appendChild(calendarBody);
+    calendarTable.appendChild(calendarFooter);
+    calendarTableContainer.appendChild(calendarTable);
+
+    var calendarSelectHour = document.createElement('select');
+    calendarSelectHour.className = 'jcalendar-select';
+    calendarSelectHour.onchange = function() {
+        obj.date[3] = this.value; 
+    }
+
+    for (var i = 0; i < 24; i++) {
+        var element = document.createElement('option');
+        element.value = i;
+        element.innerHTML = two(i);
+        calendarSelectHour.appendChild(element);
+    }
+
+    var calendarSelectMin = document.createElement('select');
+    calendarSelectMin.className = 'jcalendar-select';
+    calendarSelectMin.onchange = function() {
+        obj.date[4] = this.value; 
+    }
+
+    for (var i = 0; i < 60; i++) {
+        var element = document.createElement('option');
+        element.value = i;
+        element.innerHTML = two(i);
+        calendarSelectMin.appendChild(element);
+    }
+
+    // Footer controls
+    var calendarControlsFooter = document.createElement('div');
+    calendarControlsFooter.className = 'jcalendar-controls';
+
+    var calendarControlsTime = document.createElement('div');
+    calendarControlsTime.className = 'jcalendar-time';
+    calendarControlsTime.style.maxWidth = '140px';
+    calendarControlsTime.appendChild(calendarSelectHour);
+    calendarControlsTime.appendChild(calendarSelectMin);
+
+    var calendarControlsUpdateButton = document.createElement('input');
+    calendarControlsUpdateButton.setAttribute('type', 'button');
+    calendarControlsUpdateButton.className = 'jcalendar-update';
+    calendarControlsUpdateButton.value = 'Update';
+
+    var calendarControlsUpdate = document.createElement('div');
+    calendarControlsUpdate.style.flexGrow = '10';
+    calendarControlsUpdate.appendChild(calendarControlsUpdateButton);
+    calendarControlsFooter.appendChild(calendarControlsTime);
+    calendarControlsFooter.appendChild(calendarControlsUpdate);
+    calendarContent.appendChild(calendarControlsFooter);
+
+    var calendarBackdrop = document.createElement('div');
+    calendarBackdrop.className = 'jcalendar-backdrop';
+    calendar.appendChild(calendarBackdrop);
+
+    // Update actions button
+    var updateActions = function() {
+        var currentDay = calendar.querySelector('.jcalendar-selected');
+
+        if (currentDay && currentDay.classList.contains('jcalendar-disabled')) {
+            calendarControlsUpdateButton.setAttribute('disabled', 'disabled');
+            calendarSelectHour.setAttribute('disabled', 'disabled');
+            calendarSelectMin.setAttribute('disabled', 'disabled');
+        } else {
+            calendarControlsUpdateButton.removeAttribute('disabled');
+            calendarSelectHour.removeAttribute('disabled');
+            calendarSelectMin.removeAttribute('disabled');
+        }
+    }
+
+    // Methods
+    obj.open = function (value) {
+        if (! calendar.classList.contains('jcalendar-focus')) {
+            if (jSuites.calendar.current) {
+                jSuites.calendar.current.close();
+            }
+            // Current
+            jSuites.calendar.current = obj;
+            // Show calendar
+            calendar.classList.add('jcalendar-focus');
+            // Get days
+            obj.getDays();
+            // Hour
+            if (obj.options.time) {
+                calendarSelectHour.value = obj.date[3];
+                calendarSelectMin.value = obj.date[4];
+            }
+
+            // Get the position of the corner helper
+            if (jSuites.getWindowWidth() < 800 || obj.options.fullscreen) {
+                // Full
+                calendar.classList.add('jcalendar-fullsize');
+                // Animation
+                jSuites.slideBottom(calendarContent, 1);
+            } else {
+                const rect = el.getBoundingClientRect();
+                const rectContent = calendarContent.getBoundingClientRect();
+
+                if (obj.options.position) {
+                    calendarContainer.style.position = 'fixed';
+                    if (window.innerHeight < rect.bottom + rectContent.height) {
+                        calendarContainer.style.top = (rect.top - (rectContent.height + 2)) + 'px';
+                    } else {
+                        calendarContainer.style.top = (rect.top + rect.height + 2) + 'px';
+                    }
+                    calendarContainer.style.left = rect.left + 'px';
+                } else {
+                    if (window.innerHeight < rect.bottom + rectContent.height) {
+                        calendarContainer.style.bottom = (1 * rect.height + rectContent.height + 2) + 'px';
+                    } else {
+                        calendarContainer.style.top = 2 + 'px'; 
+                    }
+                }
+            }
+        }
+    }
+
+    obj.close = function (ignoreEvents, update) {
+        if (jSuites.calendar.current) {
+            // Current
+            jSuites.calendar.current = null;
+
+            if (update !== false) {
+                var element = calendar.querySelector('.jcalendar-selected');
+
+                if (typeof(update) == 'string') {
+                    var value = update;
+                } else if (element && element.classList.contains('jcalendar-disabled')) {
+                    var value = obj.options.value
+                } else {
+                    var value = obj.getValue();
+                }
+
+                obj.setValue(value);
+            }
+
+            // Events
+            if (! ignoreEvents && typeof(obj.options.onclose) == 'function') {
+                obj.options.onclose(el);
+            }
+
+            // Hide
+            calendar.classList.remove('jcalendar-focus');
+        }
+
+        return obj.options.value;
+    }
+
+    obj.prev = function() {
+        // Check if the visualization is the days picker or years picker
+        if (obj.options.mode == 'years') {
+            obj.date[0] = obj.date[0] - 12;
+
+            // Update picker table of days
+            obj.getYears();
+        } else {
+            // Go to the previous month
+            if (obj.date[1] < 2) {
+                obj.date[0] = obj.date[0] - 1;
+                obj.date[1] = 12;
+            } else {
+                obj.date[1] = obj.date[1] - 1;
+            }
+
+            // Update picker table of days
+            obj.getDays();
+        }
+    }
+
+    obj.next = function() {
+        // Check if the visualization is the days picker or years picker
+        if (obj.options.mode == 'years') {
+            obj.date[0] = parseInt(obj.date[0]) + 12;
+
+            // Update picker table of days
+            obj.getYears();
+        } else {
+            // Go to the previous month
+            if (obj.date[1] > 11) {
+                obj.date[0] = parseInt(obj.date[0]) + 1;
+                obj.date[1] = 1;
+            } else {
+                obj.date[1] = parseInt(obj.date[1]) + 1;
+            }
+
+            // Update picker table of days
+            obj.getDays();
+        }
+    }
+
+    obj.setValue = function(val) {
+        if (! val) {
+            val = '' + val;
+        }
+        // Values
+        var newValue = val;
+        var oldValue = obj.options.value;
+        // Set label
+        var value = obj.setLabel(newValue, obj.options.format);
+        var date = newValue.split(' ');
+        if (! date[1]) {
+            date[1] = '00:00:00';
+        }
+        var time = date[1].split(':')
+        var date = date[0].split('-');
+        var y = parseInt(date[0]);
+        var m = parseInt(date[1]);
+        var d = parseInt(date[2]);
+        var h = parseInt(time[0]);
+        var i = parseInt(time[1]);
+        obj.date = [ y, m, d, h, i, 0 ];
+        var val = obj.setLabel(newValue, obj.options.format);
+
+        if (oldValue != newValue) {
+            // Input value
+            if (el.tagName == 'INPUT') {
+                el.value = val;
+            }
+            // New value
+            obj.options.value = newValue;
+            // On change
+            if (typeof(obj.options.onchange) ==  'function') {
+                obj.options.onchange(el, newValue, oldValue);
+            }
+        }
+
+        obj.getDays();
+    }
+
+    obj.getValue = function() {
+        if (obj.date) {
+            if (obj.options.time) {
+                return two(obj.date[0]) + '-' + two(obj.date[1]) + '-' + two(obj.date[2]) + ' ' + two(obj.date[3]) + ':' + two(obj.date[4]) + ':' + two(0);
+            } else {
+                return two(obj.date[0]) + '-' + two(obj.date[1]) + '-' + two(obj.date[2]) + ' ' + two(0) + ':' + two(0) + ':' + two(0);
+            }
+        } else {
+            return "";
+        }
+    }
+
+    /**
+     *  Calendar
+     */
+    obj.update = function(element) {
+        if (element.classList.contains('jcalendar-disabled')) {
+            // Do nothing
+        } else {
+            obj.date[2] = element.innerText;
+
+            if (! obj.options.time) {
+                obj.close();
+            } else {
+                obj.date[3] = calendarSelectHour.value;
+                obj.date[4] = calendarSelectMin.value;
+            }
+
+            var elements = calendar.querySelector('.jcalendar-selected');
+            if (elements) {
+                elements.classList.remove('jcalendar-selected');
+            }
+            element.classList.add('jcalendar-selected');
+        }
+
+        // Update
+        updateActions();
+    }
+
+    /**
+     * Set to blank
+     */
+    obj.reset = function() {
+        // Close calendar
+        obj.close(false, '');
+    }
+
+    /**
+     * Get calendar days
+     */
+    obj.getDays = function() {
+        // Mode
+        obj.options.mode = 'days';
+
+        // Setting current values in case of NULLs
+        var date = new Date();
+
+        // Current selection
+        var year = obj.date && obj.date[0] ? obj.date[0] : parseInt(date.getFullYear());
+        var month = obj.date && obj.date[1] ? obj.date[1] : parseInt(date.getMonth()) + 1;
+        var day = obj.date && obj.date[2] ? obj.date[2] : parseInt(date.getDay());
+        var hour = obj.date && obj.date[3] ? obj.date[3] : parseInt(date.getHours());
+        var min = obj.date && obj.date[4] ? obj.date[4] : parseInt(date.getMinutes());
+
+        // Selection container
+        obj.date = [year, month, day, hour, min, 0 ];
+
+        // Update title
+        calendarLabelYear.innerHTML = year;
+        calendarLabelMonth.innerHTML = obj.options.months[month - 1];
+
+        // Current month and Year
+        var isCurrentMonthAndYear = (date.getMonth() == month - 1) && (date.getFullYear() == year) ? true : false;
+        var currentDay = date.getDate();
+
+        // Number of days in the month
+        var date = new Date(year, month, 0, 0, 0);
+        var numberOfDays = date.getDate();
+
+        // First day
+        var date = new Date(year, month-1, 0, 0, 0);
+        var firstDay = date.getDay() + 1;
+
+        // Index value
+        var index = obj.options.startingDay || 0;
+
+        // First of day relative to the starting calendar weekday
+        firstDay = firstDay - index;
+
+        // Reset table
+        calendarBody.innerHTML = '';
+
+        // Weekdays Row
+        var row = document.createElement('tr');
+        row.setAttribute('align', 'center');
+        calendarBody.appendChild(row);
+
+        // Create weekdays row
+        for (var i = 0; i < 7; i++) {
+            var cell = document.createElement('td');
+            cell.classList.add('jcalendar-weekday')
+            cell.innerHTML = obj.options.weekdays_short[index];
+            row.appendChild(cell);
+            // Next week day
+            index++;
+            // Restart index
+            if (index > 6) {
+                index = 0;
+            }
+        }
+
+        // Index of days
+        var index = 0;
+        var d = 0;
+ 
+        // Calendar table
+        for (var j = 0; j < 5; j++) {
+            // Reset cells container
+            var row = document.createElement('tr');
+            row.setAttribute('align', 'center');
+            // Data control
+            var emptyRow = true;
+            // Create cells
+            for (var i = 0; i < 7; i++) {
+                // Create cell
+                var cell = document.createElement('td');
+                cell.classList.add('jcalendar-set-day');
+
+                if (index >= firstDay && index < (firstDay + numberOfDays)) {
+                    // Day cell
+                    d++;
+                    cell.innerHTML = d;
+
+                    // Selected
+                    if (d == day) {
+                        cell.classList.add('jcalendar-selected');
+                    }
+
+                    // Current selection day is today
+                    if (isCurrentMonthAndYear && currentDay == d) {
+                        cell.style.fontWeight = 'bold';
+                    }
+
+                    // Current selection day
+                    var current = jSuites.calendar.now(new Date(year, month-1, d), true);
+
+                    // Available ranges
+                    if (obj.options.validRange) {
+                        if (! obj.options.validRange[0] || current >= obj.options.validRange[0]) {
+                            var test1 = true;
+                        } else {
+                            var test1 = false;
+                        }
+
+                        if (! obj.options.validRange[1] || current <= obj.options.validRange[1]) {
+                            var test2 = true;
+                        } else {
+                            var test2 = false;
+                        }
+
+                        if (! (test1 && test2)) {
+                            cell.classList.add('jcalendar-disabled');
+                        }
+                    }
+
+                    // Control
+                    emptyRow = false;
+                }
+                // Day cell
+                row.appendChild(cell);
+                // Index
+                index++;
+            }
+
+            // Add cell to the calendar body
+            if (emptyRow == false) {
+                calendarBody.appendChild(row);
+            }
+        }
+
+        // Show time controls
+        if (obj.options.time) {
+            calendarControlsTime.style.display = '';
+        } else {
+            calendarControlsTime.style.display = 'none';
+        }
+
+        // Update
+        updateActions();
+    }
+
+    obj.getMonths = function() {
+        // Mode
+        obj.options.mode = 'months';
+
+        // Loading month labels
+        var months = obj.options.months;
+
+        // Update title
+        calendarLabelYear.innerHTML = obj.date[0];
+        calendarLabelMonth.innerHTML = '';
+
+        // Create months table
+        var html = '<td colspan="7"><table width="100%"><tr align="center">';
+
+        for (i = 0; i < 12; i++) {
+            if ((i > 0) && (!(i % 4))) {
+                html += '</tr><tr align="center">';
+            }
+
+            var month = parseInt(i) + 1;
+            html += '<td class="jcalendar-set-month" data-value="' + month + '">' + months[i] +'</td>';
+        }
+
+        html += '</tr></table></td>';
+
+        calendarBody.innerHTML = html;
+    }
+
+    obj.getYears = function() { 
+        // Mode
+        obj.options.mode = 'years';
+
+        // Array of years
+        var y = [];
+        for (i = 0; i < 25; i++) {
+            y[i] = parseInt(obj.date[0]) + (i - 12);
+        }
+
+        // Assembling the year tables
+        var html = '<td colspan="7"><table width="100%"><tr align="center">';
+
+        for (i = 0; i < 25; i++) {
+            if ((i > 0) && (!(i % 5))) {
+                html += '</tr><tr align="center">';
+            }
+            html += '<td class="jcalendar-set-year">'+ y[i] +'</td>';
+        }
+
+        html += '</tr></table></td>';
+
+        calendarBody.innerHTML = html;
+    }
+
+    obj.setLabel = function(value, format) {
+        return jSuites.calendar.getDateString(value, format);
+    }
+
+    obj.fromFormatted = function (value, format) {
+        return jSuites.calendar.extractDateFromString(value, format);
+    }
+
+    var mouseUpControls = function(e) {
+        var action = e.target.className;
+
+        // Object id
+        if (action == 'jcalendar-prev') {
+            obj.prev();
+            e.stopPropagation();
+            e.preventDefault();
+        } else if (action == 'jcalendar-next') {
+            obj.next();
+            e.stopPropagation();
+            e.preventDefault();
+        } else if (action == 'jcalendar-month') {
+            obj.getMonths();
+            e.stopPropagation();
+            e.preventDefault();
+        } else if (action == 'jcalendar-year') {
+            obj.getYears();
+            e.stopPropagation();
+            e.preventDefault();
+        } else if (action == 'jcalendar-set-year') {
+            obj.date[0] = e.target.innerText;
+            obj.getDays();
+            e.stopPropagation();
+            e.preventDefault();
+        } else if (action == 'jcalendar-set-month') {
+            obj.date[1] = parseInt(e.target.getAttribute('data-value'));
+            obj.getDays();
+            e.stopPropagation();
+            e.preventDefault();
+        } else if (action == 'jcalendar-confirm' || action == 'jcalendar-update') {
+            obj.close();
+            e.stopPropagation();
+            e.preventDefault();
+        } else if (action == 'jcalendar-close') {
+            obj.close();
+            e.stopPropagation();
+            e.preventDefault();
+        } else if (action == 'jcalendar-backdrop') {
+            obj.close(false, false);
+            e.stopPropagation();
+            e.preventDefault();
+        } else if (action == 'jcalendar-reset') {
+            obj.reset();
+            e.stopPropagation();
+            e.preventDefault();
+        } else if (e.target.classList.contains('jcalendar-set-day')) {
+            if (e.target.innerText) {
+                obj.update(e.target);
+                e.stopPropagation();
+                e.preventDefault();
+            }
+        }
+    }
+
+    var keyUpControls = function(e) {
+        if (e.target.value && e.target.value.length > 3) {
+            var test = jSuites.calendar.extractDateFromString(e.target.value, obj.options.format);
+            if (test) {
+                if (e.target.getAttribute('data-completed') == 'true') {
+                    obj.setValue(test);
+                }
+            }
+        }
+    }
+
+    // Handle events
+    el.addEventListener("keyup", keyUpControls);
+
+    // Add global events
+    calendar.addEventListener("swipeleft", function(e) {
+        jSuites.slideLeft(calendarTable, 0, function() {
+            obj.next();
+            jSuites.slideRight(calendarTable, 1);
+        });
+        e.preventDefault();
+        e.stopPropagation();
+    });
+
+    calendar.addEventListener("swiperight", function(e) {
+        jSuites.slideRight(calendarTable, 0, function() {
+            obj.prev();
+            jSuites.slideLeft(calendarTable, 1);
+        });
+        e.preventDefault();
+        e.stopPropagation();
+    });
+
+    if ('ontouchend' in document.documentElement === true) {
+        calendar.addEventListener("touchend", mouseUpControls);
+
+        el.addEventListener("touchend", function(e) {
+            obj.open();
+        });
+    } else {
+        calendar.addEventListener("mouseup", mouseUpControls);
+
+        el.addEventListener("mouseup", function(e) {
+            obj.open();
+        });
+    }
+
+    // Append element to the DOM
+    if (el.tagName == 'INPUT') {
+        el.parentNode.insertBefore(calendar, el.nextSibling);
+        // Add properties
+        el.setAttribute('autocomplete', 'off');
+        el.setAttribute('data-mask', obj.options.format.toLowerCase());
+
+        if (obj.options.readonly) {
+            el.setAttribute('readonly', 'readonly');
+        }
+        if (obj.options.placeholder) {
+            el.setAttribute('placeholder', obj.options.placeholder);
+        }
+        // Element
+        el.classList.add('jcalendar-input');
+        // Value
+        el.value = obj.setLabel(obj.getValue(), obj.options.format);
+    }
+
+    // Keep object available from the node
+    el.calendar = obj;
+
+    if (obj.options.opened == true) {
+        obj.open();
+    }
+
+    return obj;
+});
+
+jSuites.calendar.prettify = function(d, texts) {
+    if (! texts) {
+        var texts = {
+            justNow: 'Just now',
+            xMinutesAgo: '{0}m ago',
+            xHoursAgo: '{0}h ago',
+            xDaysAgo: '{0}d ago',
+            xWeeksAgo: '{0}w ago',
+            xMonthsAgo: '{0} mon ago',
+            xYearsAgo: '{0}y ago',
+        }
+    }
+
+    var d1 = new Date();
+    var d2 = new Date(d);
+    var total = parseInt((d1 - d2) / 1000 / 60);
+
+    String.prototype.format = function(o) {
+        return this.replace('{0}', o);
+    }
+
+    if (total == 0) {
+        var text = texts.justNow;
+    } else if (total < 90) {
+        var text = texts.xMinutesAgo.format(total);
+    } else if (total < 1440) { // One day
+        var text = texts.xHoursAgo.format(Math.round(total/60));
+    } else if (total < 20160) { // 14 days
+        var text = texts.xDaysAgo.format(Math.round(total / 1440));
+    } else if (total < 43200) { // 30 days
+        var text = texts.xWeeksAgo.format(Math.round(total / 10080));
+    } else if (total < 1036800) { // 24 months
+        var text = texts.xMonthsAgo.format(Math.round(total / 43200));
+    } else { // 24 months+
+        var text = texts.xYearsAgo.format(Math.round(total / 525600));
+    }
+
+    return text;
+}
+
+jSuites.calendar.prettifyAll = function() {
+    var elements = document.querySelectorAll('.prettydate');
+    for (var i = 0; i < elements.length; i++) {
+        if (elements[i].getAttribute('data-date')) {
+            elements[i].innerHTML = jSuites.calendar.prettify(elements[i].getAttribute('data-date'));
+        } else {
+            elements[i].setAttribute('data-date', elements[i].innerHTML);
+            elements[i].innerHTML = jSuites.calendar.prettify(elements[i].innerHTML);
+        }
+    }
+}
+
+jSuites.calendar.now = function(date, dateOnly) {
+    if (! date) {
+        var date = new Date();
+    }
+    var y = date.getFullYear();
+    var m = date.getMonth() + 1;
+    var d = date.getDate();
+    var h = date.getHours();
+    var i = date.getMinutes();
+    var s = date.getSeconds();
+
+    // Two digits
+    var two = function(value) {
+        value = '' + value;
+        if (value.length == 1) {
+            value = '0' + value;
+        }
+        return value;
+    }
+
+    if (dateOnly == true) {
+        return two(y) + '-' + two(m) + '-' + two(d);
+    } else {
+        return two(y) + '-' + two(m) + '-' + two(d) + ' ' + two(h) + ':' + two(i) + ':' + two(s);
+    }
+}
+
+// Helper to extract date from a string
+jSuites.calendar.extractDateFromString = function(date, format) {
+    format = "DD/MM/YYYY";
+    var v1 = '' + date;
+    var v2 = format.replace(/[0-9]/g,'');
+
+    var test = 1;
+
+    // Get year
+    var y = v2.search("YYYY");
+    y = v1.substr(y,4);
+    if (parseInt(y) != y) {
+        test = 0;
+    }
+
+    // Get month
+    var m = v2.search("MM");
+    m = v1.substr(m,2);
+    if (parseInt(m) != m || d > 12) {
+        test = 0;
+    }
+
+    // Get day
+    var d = v2.search("DD");
+    d = v1.substr(d,2);
+    if (parseInt(d) != d  || d > 31) {
+        test = 0;
+    }
+
+    // Get hour
+    var h = v2.search("HH");
+    if (h >= 0) {
+        h = v1.substr(h,2);
+        if (! parseInt(h) || h > 23) {
+            h = '00';
+        }
+    } else {
+        h = '00';
+    }
+    
+    // Get minutes
+    var i = v2.search("MI");
+    if (i >= 0) {
+        i = v1.substr(i,2);
+        if (! parseInt(i) || i > 59) {
+            i = '00';
+        }
+    } else {
+        i = '00';
+    }
+
+    // Get seconds
+    var s = v2.search("SS");
+    if (s >= 0) {
+        s = v1.substr(s,2);
+        if (! parseInt(s) || s > 59) {
+            s = '00';
+        }
+    } else {
+        s = '00';
+    }
+
+    if (test == 1 && date.length == v2.length) {
+        // Update source
+        var data = y + '-' + m + '-' + d + ' ' + h + ':' +  i + ':' + s;
+
+        return data;
+    }
+
+    return '';
+}
+
+// Helper to convert date into string
+jSuites.calendar.getDateString = function(value, format) {
+    // Default calendar
+    if (! format) {
+        var format = 'DD/MM/YYYY';
+    }
+
+    if (value) {
+        var d = ''+value;
+        d = d.split(' ');
+
+        var h = '';
+        var m = '';
+        var s = '';
+
+        if (d[1]) {
+            h = d[1].split(':');
+            m = h[1] ? h[1] : '00';
+            s = h[2] ? h[2] : '00';
+            h = h[0] ? h[0] : '00';
+        } else {
+            h = '00';
+            m = '00';
+            s = '00';
+        }
+
+        d = d[0].split('-');
+
+        if (d[0] && d[1] && d[2] && d[0] > 0 && d[1] > 0 && d[1] < 13 && d[2] > 0 && d[2] < 32) {
+            var calendar = new Date(d[0], d[1]-1, d[2]);
+            var weekday = new Array('Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday');
+            var months = new Array('Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec');
+
+            d[1] = (d[1].length < 2 ? '0' : '') + d[1];
+            d[2] = (d[2].length < 2 ? '0' : '') + d[2];
+            h = (h.length < 2 ? '0' : '') + h;
+            m = (m.length < 2 ? '0' : '') + m;
+            s = (s.length < 2 ? '0' : '') + s;
+
+            value = format;
+            value = value.replace('WD', weekday[calendar.getDay()]);
+            value = value.replace('DD', d[2]);
+            value = value.replace('MM', d[1]);
+            value = value.replace('YYYY', d[0]);
+            value = value.replace('YY', d[0].substring(2,4));
+            value = value.replace('MON', months[parseInt(d[1])-1].toUpperCase());
+
+            if (h) {
+                value = value.replace('HH24', h);
+            }
+
+            if (h > 12) {
+                value = value.replace('HH12', h - 12);
+                value = value.replace('HH', h);
+            } else {
+                value = value.replace('HH12', h);
+                value = value.replace('HH', h);
+            }
+
+            value = value.replace('MI', m);
+            value = value.replace('MM', m);
+            value = value.replace('SS', s);
+        } else {
+            value = '';
+        }
+    }
+
+    return value;
+}
+
+jSuites.calendar.isOpen = function(e) {
+    if (jSuites.calendar.current) {
+        if (! e.target.className || e.target.className.indexOf('jcalendar') == -1) {
+            jSuites.calendar.current.close(false, false);
+        }
+    }
+}
+
+if ('ontouchstart' in document.documentElement === true) {
+    document.addEventListener("touchstart", jSuites.calendar.isOpen);
+} else {
+    document.addEventListener("mousedown", jSuites.calendar.isOpen);
+}
+
+jSuites.color = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+    obj.values = [];
+
+    // Global container
+    if (! jSuites.color.current) {
+        jSuites.color.current = null;
+    }
+
+    /**
+     * @typedef {Object} defaults
+     * @property {(string|Array)} value - Initial value of the compontent
+     * @property {string} placeholder - The default instruction text on the element
+     * @property {requestCallback} onchange - Method to be execute after any changes on the element
+     * @property {requestCallback} onclose - Method to be execute when the element is closed
+     */
+    var defaults = {
+        placeholder: '',
+        value: null,
+        onclose: null,
+        onchange: null,
+        closeOnChange: true,
+    };
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    var palette = {
+          "red": {
+            "50": "#ffebee",
+            "100": "#ffcdd2",
+            "200": "#ef9a9a",
+            "300": "#e57373",
+            "400": "#ef5350",
+            "500": "#f44336",
+            "600": "#e53935",
+            "700": "#d32f2f",
+            "800": "#c62828",
+            "900": "#b71c1c",
+          },
+          "pink": {
+            "50": "#fce4ec",
+            "100": "#f8bbd0",
+            "200": "#f48fb1",
+            "300": "#f06292",
+            "400": "#ec407a",
+            "500": "#e91e63",
+            "600": "#d81b60",
+            "700": "#c2185b",
+            "800": "#ad1457",
+            "900": "#880e4f",
+          },
+          "purple": {
+            "50": "#f3e5f5",
+            "100": "#e1bee7",
+            "200": "#ce93d8",
+            "300": "#ba68c8",
+            "400": "#ab47bc",
+            "500": "#9c27b0",
+            "600": "#8e24aa",
+            "700": "#7b1fa2",
+            "800": "#6a1b9a",
+            "900": "#4a148c",
+          },
+          "indigo": {
+            "50": "#e8eaf6",
+            "100": "#c5cae9",
+            "200": "#9fa8da",
+            "300": "#7986cb",
+            "400": "#5c6bc0",
+            "500": "#3f51b5",
+            "600": "#3949ab",
+            "700": "#303f9f",
+            "800": "#283593",
+            "900": "#1a237e",
+          },
+          "blue": {
+            "50": "#e3f2fd",
+            "100": "#bbdefb",
+            "200": "#90caf9",
+            "300": "#64b5f6",
+            "400": "#42a5f5",
+            "500": "#2196f3",
+            "600": "#1e88e5",
+            "700": "#1976d2",
+            "800": "#1565c0",
+            "900": "#0d47a1",
+          },
+          "cyan": {
+            "50": "#e0f7fa",
+            "100": "#b2ebf2",
+            "200": "#80deea",
+            "300": "#4dd0e1",
+            "400": "#26c6da",
+            "500": "#00bcd4",
+            "600": "#00acc1",
+            "700": "#0097a7",
+            "800": "#00838f",
+            "900": "#006064",
+          },
+          "teal": {
+            "50": "#e0f2f1",
+            "100": "#b2dfdb",
+            "200": "#80cbc4",
+            "300": "#4db6ac",
+            "400": "#26a69a",
+            "500": "#009688",
+            "600": "#00897b",
+            "700": "#00796b",
+            "800": "#00695c",
+            "900": "#004d40",
+          },
+          "green": {
+            "50": "#e8f5e9",
+            "100": "#c8e6c9",
+            "200": "#a5d6a7",
+            "300": "#81c784",
+            "400": "#66bb6a",
+            "500": "#4caf50",
+            "600": "#43a047",
+            "700": "#388e3c",
+            "800": "#2e7d32",
+            "900": "#1b5e20",
+          },
+          "lightgreen": {
+            "50": "#f1f8e9",
+            "100": "#dcedc8",
+            "200": "#c5e1a5",
+            "300": "#aed581",
+            "400": "#9ccc65",
+            "500": "#8bc34a",
+            "600": "#7cb342",
+            "700": "#689f38",
+            "800": "#558b2f",
+            "900": "#33691e",
+          },
+          "lime": {
+            "50": "#f9fbe7",
+            "100": "#f0f4c3",
+            "200": "#e6ee9c",
+            "300": "#dce775",
+            "400": "#d4e157",
+            "500": "#cddc39",
+            "600": "#c0ca33",
+            "700": "#afb42b",
+            "800": "#9e9d24",
+            "900": "#827717",
+          },
+          "yellow": {
+            "50": "#fffde7",
+            "100": "#fff9c4",
+            "200": "#fff59d",
+            "300": "#fff176",
+            "400": "#ffee58",
+            "500": "#ffeb3b",
+            "600": "#fdd835",
+            "700": "#fbc02d",
+            "800": "#f9a825",
+            "900": "#f57f17",
+          },
+          "amber": {
+            "50": "#fff8e1",
+            "100": "#ffecb3",
+            "200": "#ffe082",
+            "300": "#ffd54f",
+            "400": "#ffca28",
+            "500": "#ffc107",
+            "600": "#ffb300",
+            "700": "#ffa000",
+            "800": "#ff8f00",
+            "900": "#ff6f00",
+          },
+          "orange": {
+            "50": "#fff3e0",
+            "100": "#ffe0b2",
+            "200": "#ffcc80",
+            "300": "#ffb74d",
+            "400": "#ffa726",
+            "500": "#ff9800",
+            "600": "#fb8c00",
+            "700": "#f57c00",
+            "800": "#ef6c00",
+            "900": "#e65100",
+          },
+          "deeporange": {
+            "50": "#fbe9e7",
+            "100": "#ffccbc",
+            "200": "#ffab91",
+            "300": "#ff8a65",
+            "400": "#ff7043",
+            "500": "#ff5722",
+            "600": "#f4511e",
+            "700": "#e64a19",
+            "800": "#d84315",
+            "900": "#bf360c",
+          },
+          "brown": {
+            "50": "#efebe9",
+            "100": "#d7ccc8",
+            "200": "#bcaaa4",
+            "300": "#a1887f",
+            "400": "#8d6e63",
+            "500": "#795548",
+            "600": "#6d4c41",
+            "700": "#5d4037",
+            "800": "#4e342e",
+            "900": "#3e2723"
+          },
+          "grey": {
+            "50": "#fafafa",
+            "100": "#f5f5f5",
+            "200": "#eeeeee",
+            "300": "#e0e0e0",
+            "400": "#bdbdbd",
+            "500": "#9e9e9e",
+            "600": "#757575",
+            "700": "#616161",
+            "800": "#424242",
+            "900": "#212121"
+          },
+          "bluegrey": {
+            "50": "#eceff1",
+            "100": "#cfd8dc",
+            "200": "#b0bec5",
+            "300": "#90a4ae",
+            "400": "#78909c",
+            "500": "#607d8b",
+            "600": "#546e7a",
+            "700": "#455a64",
+            "800": "#37474f",
+            "900": "#263238"
+          }
+    };
+
+    var x = 0;
+    var y = 0;
+    var colors = [];
+
+    var col = Object.keys(palette);
+    var shade = Object.keys(palette[col[0]]);
+
+    for (var i = 0; i < col.length; i++) {
+        for (var j = 0; j < shade.length; j++) {
+            if (! colors[j]) {
+                colors[j] = [];
+            }
+            colors[j][i] = palette[col[i]][shade[j]];
+        }
+    };
+
+    // Value
+    if (obj.options.value) {
+        el.value = obj.options.value;
+    }
+
+    // Table container
+    var container = document.createElement('div');
+    container.className = 'jcolor';
+
+    // Table container
+    var backdrop = document.createElement('div');
+    backdrop.className = 'jcolor-backdrop';
+    container.appendChild(backdrop);
+
+    // Content
+    var content = document.createElement('div');
+    content.className = 'jcolor-content';
+
+    // Close button
+    var closeButton  = document.createElement('div');
+    closeButton.className = 'jcolor-close';
+    closeButton.innerHTML = 'Done';
+    closeButton.onclick = function() {
+        obj.close();
+    }
+    content.appendChild(closeButton);
+
+    // Table pallete
+    var table = document.createElement('table');
+    table.setAttribute('cellpadding', '7');
+    table.setAttribute('cellspacing', '0');
+
+    for (var i = 0; i < colors.length; i++) {
+        var tr = document.createElement('tr');
+        for (var j = 0; j < colors[i].length; j++) {
+            var td = document.createElement('td');
+            td.style.backgroundColor = colors[i][j];
+            td.setAttribute('data-value', colors[i][j]);
+            td.innerHTML = '';
+            tr.appendChild(td);
+
+            // Selected color
+            if (obj.options.value == colors[i][j]) {
+                td.classList.add('jcolor-selected');
+            }
+
+            // Possible values
+            obj.values[colors[i][j]] = td;
+        }
+        table.appendChild(tr);
+    }
+
+    /**
+     * Open color pallete
+     */
+    obj.open = function() {
+        if (jSuites.color.current) {
+            if (jSuites.color.current != obj) {
+                jSuites.color.current.close();
+            }
+        }
+
+        if (! jSuites.color.current) {
+            // Persist element
+            jSuites.color.current = obj;
+            // Show colorpicker
+            container.classList.add('jcolor-focus');
+
+            const rectContent = content.getBoundingClientRect();
+
+            if (jSuites.getWindowWidth() < 800) {
+                content.style.top = '';
+                content.classList.add('jcolor-fullscreen');
+                jSuites.slideBottom(content, 1);
+                backdrop.style.display = 'block';
+            } else {
+                if (content.classList.contains('jcolor-fullscreen')) {
+                    content.classList.remove('jcolor-fullscreen');
+                    backdrop.style.display = '';
+                }
+
+                const rect = el.getBoundingClientRect();
+
+                if (window.innerHeight < rect.bottom + rectContent.height) {
+                    content.style.top = -1 * (rectContent.height + rect.height + 2) + 'px';
+                } else {
+                    content.style.top = '2px';
+                }
+            }
+
+            container.focus();
+        }
+    }
+
+    /**
+     * Close color pallete
+     */
+    obj.close = function(ignoreEvents) {
+        if (jSuites.color.current) {
+            jSuites.color.current = null;
+            if (! ignoreEvents && typeof(obj.options.onclose) == 'function') {
+                obj.options.onclose(el);
+            }
+            container.classList.remove('jcolor-focus');
+        }
+
+        // Make sure backdrop is hidden
+        backdrop.style.display = '';
+
+        return obj.options.value;
+    }
+
+    /**
+     * Set value
+     */
+    obj.setValue = function(color) {
+        if (color) {
+            el.value = color;
+            obj.options.value = color;
+        }
+
+        // Remove current selecded mark
+        var selected = container.querySelector('.jcolor-selected');
+        if (selected) {
+            selected.classList.remove('jcolor-selected');
+        }
+
+        // Mark cell as selected
+        obj.values[color].classList.add('jcolor-selected');
+
+        // Onchange
+        if (typeof(obj.options.onchange) == 'function') {
+            obj.options.onchange(el, color);
+        }
+
+        if (obj.options.closeOnChange == true) {
+            obj.close();
+        }
+    }
+
+    /**
+     * Get value
+     */
+    obj.getValue = function() {
+        return obj.options.value;
+    }
+
+    /**
+     * If element is focus open the picker
+     */
+    el.addEventListener("focus", function(e) {
+        obj.open();
+    });
+
+    el.addEventListener("mousedown", function(e) {
+        if (! jSuites.color.current) {
+            setTimeout(function() {
+        obj.open();
+                e.preventDefault();
+            }, 200);
+        }
+    });
+
+    // Select color
+    container.addEventListener("mouseup", function(e) {
+        if (e.target.tagName == 'TD') {
+            jSuites.color.current.setValue(e.target.getAttribute('data-value'));
+
+            if (jSuites.color.current) {
+                jSuites.color.current.close();
+            }
+        }
+    });
+
+    // Close controller
+    document.addEventListener("mousedown", function(e) {
+        if (jSuites.color.current) {
+            var element = jSuites.getElement(e.target, 'jcolor');
+            if (! element) {
+                jSuites.color.current.close();
+            }
+        }
+    });
+
+    // Possible to focus the container
+    container.setAttribute('tabindex', '900');
+
+    // Placeholder
+    if (obj.options.placeholder) {
+        el.setAttribute('placeholder', obj.options.placeholder);
+    }
+
+    // Append to the table
+    content.appendChild(table);
+    container.appendChild(content);
+
+    // Insert picker after the element
+    if (el.tagName == 'INPUT') {
+        el.parentNode.insertBefore(container, el.nextSibling);
+    } else {
+        el.appendChild(container);
+    }
+
+    // Keep object available from the node
+    el.color = obj;
+
+    return obj;
+});
+
+
+jSuites.contextmenu = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // Default configuration
+    var defaults = {
+        items: null,
+        onclick: null,
+    };
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Class definition
+    el.classList.add('jcontextmenu');
+    // Focusable
+    el.setAttribute('tabindex', '900');
+
+    /**
+     * Open contextmenu
+     */
+    obj.open = function(e, items) {
+        if (items) {
+            // Update content
+            obj.options.items = items;
+            // Create items
+            obj.create(items);
+        }
+        // Coordinates
+        if (e.target) {
+            var x = e.clientX;
+            var y = e.clientY;
+        } else {
+            var x = e.x;
+            var y = e.y;
+        }
+
+        el.classList.add('jcontextmenu-focus');
+        el.focus();
+
+        const rect = el.getBoundingClientRect();
+
+        if (window.innerHeight < y + rect.height) {
+            el.style.top = (y - rect.height) + 'px';
+        } else {
+            el.style.top = y + 'px';
+        }
+
+        if (window.innerWidth < x + rect.width) {
+            if (x - rect.width > 0) {
+                el.style.left = (x - rect.width) + 'px';
+            } else {
+                el.style.left = '10px';
+            }
+        } else {
+            el.style.left = x + 'px';
+        }
+    }
+
+    /**
+     * Close menu
+     */
+    obj.close = function() {
+        if (el.classList.contains('jcontextmenu-focus')) {
+            el.classList.remove('jcontextmenu-focus');
+        }
+    }
+
+    /**
+     * Create items based on the declared objectd
+     * @param {object} items - List of object
+     */
+    obj.create = function(items) {
+        // Update content
+        el.innerHTML = '';
+
+        // Append items
+        for (var i = 0; i < items.length; i++) {
+            if (items[i].type && items[i].type == 'line') {
+                var itemContainer = document.createElement('hr');
+            } else {
+                var itemContainer = document.createElement('div');
+                var itemText = document.createElement('a');
+                itemText.innerHTML = items[i].title;
+
+                if (items[i].disabled) {
+                    itemContainer.className = 'jcontextmenu-disabled';
+                } else if (items[i].onclick) {
+                    itemContainer.method = items[i].onclick;
+                    itemContainer.addEventListener("mouseup", function() {
+                        // Execute method
+                        this.method(this);
+                    });
+                }
+                itemContainer.appendChild(itemText);
+
+                if (items[i].shortcut) {
+                    var itemShortCut = document.createElement('span');
+                    itemShortCut.innerHTML = items[i].shortcut;
+                    itemContainer.appendChild(itemShortCut);
+                }
+            }
+
+            el.appendChild(itemContainer);
+        }
+    }
+
+    if (typeof(obj.options.onclick) == 'function') {
+        el.addEventListener('click', function(e) {
+            obj.options.onclick(obj);
+        });
+    }
+
+    el.addEventListener('blur', function(e) {
+        setTimeout(function() {
+            obj.close();
+        }, 120);
+    });
+
+    window.addEventListener("mousewheel", function() {
+        obj.close();
+    });
+
+    // Create items
+    if (obj.options.items) {
+        obj.create(obj.options.items);
+    }
+
+    el.contextmenu = obj;
+
+    return obj;
+});
+
+jSuites.contextmenu.getElement = function(element) {
+    var foundId = 0;
+
+    function path (element) {
+        if (element.parentNode && element.getAttribute('aria-contextmenu-id')) {
+            foundId = element.getAttribute('aria-contextmenu-id')
+        } else {
+            if (element.parentNode) {
+                path(element.parentNode);
+            }
+        }
+    }
+
+    path(element);
+
+    return foundId;
+}
+
+document.addEventListener("contextmenu", function(e) {
+    var id = jSuites.contextmenu.getElement(e.target);
+    if (id) {
+        var element = document.querySelector('#' + id);
+        if (! element) {
+            console.error('JSUITES: Contextmenu id not found');
+        } else {
+            element.contextmenu.open(e);
+            e.preventDefault();
+        }
+    }
+});
+
+/**
+ * Dialog v1.0.1
+ * Author: paul.hodel@gmail.com
+ * https://github.com/paulhodel/jtools
+ */
+ 
+jSuites.dialog = (function() {
+    var obj = {};
+    obj.options = {};
+
+    var dialog = document.createElement('div');
+    dialog.setAttribute('tabindex', '901');
+    dialog.className = 'jdialog';
+    dialog.id = 'dialog';
+
+    var dialogHeader = document.createElement('div');
+    dialogHeader.className = 'jdialog-header';
+
+    var dialogTitle = document.createElement('div');
+    dialogTitle.className = 'jdialog-title';
+    dialogHeader.appendChild(dialogTitle);
+
+    var dialogMessage = document.createElement('div');
+    dialogMessage.className = 'jdialog-message';
+    dialogHeader.appendChild(dialogMessage);
+
+    var dialogFooter = document.createElement('div');
+    dialogFooter.className = 'jdialog-footer';
+
+    var dialogContainer = document.createElement('div');
+    dialogContainer.className = 'jdialog-container';
+    dialogContainer.appendChild(dialogHeader);
+    dialogContainer.appendChild(dialogFooter);
+
+    // Confirm
+    var dialogConfirm = document.createElement('div');
+    var dialogConfirmButton = document.createElement('input');
+    dialogConfirmButton.value = obj.options.confirmLabel;
+    dialogConfirmButton.type = 'button';
+    dialogConfirmButton.onclick = function() {
+        if (typeof(obj.options.onconfirm) == 'function') {
+            obj.options.onconfirm();
+        }
+        obj.close();
+    };
+    dialogConfirm.appendChild(dialogConfirmButton);
+    dialogFooter.appendChild(dialogConfirm);
+
+    // Cancel
+    var dialogCancel = document.createElement('div');
+    var dialogCancelButton = document.createElement('input');
+    dialogCancelButton.value = obj.options.cancelLabel;
+    dialogCancelButton.type = 'button';
+    dialogCancelButton.onclick = function() {
+        if (typeof(obj.options.oncancel) == 'function') {
+            obj.options.oncancel();
+        }
+        obj.close();
+    }
+    dialogCancel.appendChild(dialogCancelButton);
+    dialogFooter.appendChild(dialogCancel);
+
+    // Dialog
+    dialog.appendChild(dialogContainer);
+
+    obj.open = function(options) {
+        obj.options = options;
+
+        if (obj.options.title) {
+            dialogTitle.innerHTML = obj.options.title;
+        }
+
+        if (obj.options.message) {
+            dialogMessage.innerHTML = obj.options.message;
+        }
+
+        if (! obj.options.confirmLabel) {
+            obj.options.confirmLabel = 'OK';
+        }
+        dialogConfirmButton.value = obj.options.confirmLabel;
+
+        if (! obj.options.cancelLabel) {
+            obj.options.cancelLabel = 'Cancel';
+        }
+        dialogCancelButton.value = obj.options.cancelLabel;
+
+        if (obj.options.type == 'confirm') {
+            dialogCancelButton.parentNode.style.display = '';
+        } else {
+            dialogCancelButton.parentNode.style.display = 'none';
+        }
+
+        // Append element to the app
+        dialog.style.opacity = 100;
+
+        // Append to the page
+        if (jSuites.el) {
+            jSuites.el.appendChild(dialog);
+        } else {
+            document.body.appendChild(dialog);
+        }
+
+        // Focus
+        dialog.focus();
+
+        // Show
+        setTimeout(function() {
+            dialogContainer.style.opacity = 100;
+        }, 0);
+    };
+
+    obj.close = function() {
+        dialog.style.opacity = 0;
+        dialogContainer.style.opacity = 0;
+        setTimeout(function() {
+            dialog.remove();
+        }, 100);
+    };
+
+    return obj;
+})();
+
+jSuites.confirm = (function(message, onconfirm) {
+    if (jSuites.getWindowWidth() < 800) {
+        jSuites.dialog.open({
+            type: 'confirm',
+            message: message,
+            title: 'Confirmation',
+            onconfirm: onconfirm,
+        });
+    } else {
+        if (confirm(message)) {
+            onconfirm();
+        }
+    }
+});
+
+jSuites.alert = function(message) {
+    if (jSuites.getWindowWidth() < 800) {
+        jSuites.dialog.open({
+            title:'Alert',
+            message:message,
+        });
+    } else {
+        alert(message);
+    }
+}
+
+
+jSuites.dropdown = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // If the element is a SELECT tag, create a configuration object
+    if (el.tagName == 'SELECT') {
+        var ret = jSuites.dropdown.extractFromDom(el, options);
+        el = ret.el;
+        options = ret.options;
+    }
+
+    // Default configuration
+    var defaults = {
+        url: null,
+        data: [],
+        multiple: false,
+        autocomplete: false,
+        type: null,
+        width: null,
+        opened: false,
+        value: null,
+        placeholder: '',
+        position: false,
+        onchange: null,
+        onload: null,
+        onopen: null,
+        onclose: null,
+        onblur: null,
+    };
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Global container
+    if (! jSuites.dropdown.current) {
+        jSuites.dropdown.current = null;
+    }
+
+    // Containers
+    obj.items = [];
+    obj.groups = [];
+    obj.selected = [];
+
+    // Create dropdown
+    el.classList.add('jdropdown');
+ 
+    if (obj.options.type == 'searchbar') {
+        el.classList.add('jdropdown-searchbar');
+    } else if (obj.options.type == 'list') {
+        el.classList.add('jdropdown-list');
+    } else if (obj.options.type == 'picker') {
+        el.classList.add('jdropdown-picker');
+    } else {
+        if (jSuites.getWindowWidth() < 800) {
+            el.classList.add('jdropdown-picker');
+            obj.options.type = 'picker';
+        } else {
+            if (obj.options.width) {
+                el.style.width = obj.options.width;
+                el.style.minWidth = obj.options.width;
+            }
+            el.classList.add('jdropdown-default');
+            obj.options.type = 'default';
+        }
+    }
+
+    // Header container
+    var containerHeader = document.createElement('div');
+    containerHeader.className = 'jdropdown-container-header';
+
+    // Header
+    var header = document.createElement('input');
+    header.className = 'jdropdown-header';
+    if (typeof(obj.options.onblur) == 'function') {
+        header.onblur = function() {
+            obj.options.onblur(el);
+        }
+    }
+
+    // Container
+    var container = document.createElement('div');
+    container.className = 'jdropdown-container';
+
+    // Dropdown content
+    var content = document.createElement('div');
+    content.className = 'jdropdown-content';
+
+    // Close button
+    var closeButton  = document.createElement('div');
+    closeButton.className = 'jdropdown-close';
+    closeButton.innerHTML = 'Done';
+
+    // Create backdrop
+    var backdrop  = document.createElement('div');
+    backdrop.className = 'jdropdown-backdrop';
+
+    // Autocomplete
+    if (obj.options.autocomplete == true) {
+        el.setAttribute('data-autocomplete', true);
+
+        // Handler
+        var keyTimer = null;
+        header.addEventListener('keyup', function(e) {
+            if (keyTimer) {
+                clearTimeout(keyTimer);
+            }
+            keyTimer = setTimeout(function() {
+                obj.find(header.value);
+                keyTimer = null;
+            }, 500);
+
+            if (! el.classList.contains('jdropdown-focus')) {
+                if (e.which > 65) {
+                    obj.open();
+                }
+            }
+        });
+    } else {
+        header.setAttribute('readonly', 'readonly');
+    }
+
+    // Place holder
+    if (! obj.options.placeholder && el.getAttribute('placeholder')) {
+        obj.options.placeholder = el.getAttribute('placeholder');
+    }
+
+    if (obj.options.placeholder) {
+        header.setAttribute('placeholder', obj.options.placeholder);
+    }
+
+    // Append elements
+    containerHeader.appendChild(header);
+    if (obj.options.type == 'searchbar') {
+        containerHeader.appendChild(closeButton);
+    } else {
+        container.appendChild(closeButton);
+    }
+    container.appendChild(content);
+    el.appendChild(containerHeader);
+    el.appendChild(container);
+    el.appendChild(backdrop);
+
+    /**
+     * Init dropdown
+     */
+    obj.init = function() {
+        if (obj.options.url) {
+            jSuites.ajax({
+                url: obj.options.url,
+                method: 'GET',
+                dataType: 'json',
+                success: function(data) {
+                    if (data) {
+                        // Set data
+                        obj.setData(data);
+                        // Set value
+                        if (obj.options.value != null) {
+                            obj.setValue(obj.options.value);
+                        }
+                        // Onload method
+                        if (typeof(obj.options.onload) == 'function') {
+                            obj.options.onload(el, obj, data);
+                        }
+                    }
+                }
+            });
+        } else {
+            // Set data
+            obj.setData();
+            // Set value
+            if (obj.options.value != null) {
+                obj.setValue(obj.options.value);
+            }
+            // Onload
+            if (typeof(obj.options.onload) == 'function') {
+                obj.options.onload(el, obj, data);
+            }
+        }
+
+        // Open dropdown
+        if (obj.options.opened == true) {
+            obj.open();
+        }
+    }
+
+    obj.getUrl = function() {
+        return obj.options.url;
+    }
+
+    obj.setUrl = function(url) {
+        obj.options.url = url;
+
+        jSuites.ajax({
+            url: obj.options.url,
+            method: 'GET',
+            dataType: 'json',
+            success: function(data) {
+                obj.setData(data);
+            }
+        });
+    }
+
+    /**
+     * Create a new item
+     */
+    obj.createItem = function(data) {
+        // Create item
+        var item = {};
+        item.element = document.createElement('div');
+        item.element.className = 'jdropdown-item';
+        item.value = data.id;
+        item.text = data.name;
+        item.textLowerCase = '' + data.name.toLowerCase();
+
+        // Image
+        if (data.image) {
+            var image = document.createElement('img');
+            image.className = 'jdropdown-image';
+            image.src = data.image;
+            if (! data.title) {
+               image.classList.add('jdropdown-image-small');
+            }
+            item.element.appendChild(image);
+        }
+
+        // Set content
+        var node = document.createElement('div');
+        node.className = 'jdropdown-description';
+        node.innerHTML = data.name;
+
+        // Title
+        if (data.title) {
+            var title = document.createElement('div');
+            title.className = 'jdropdown-title';
+            title.innerHTML = data.title;
+            node.appendChild(title);
+        }
+
+        // Add node to item
+        item.element.appendChild(node);
+
+        return item;
+    }
+
+    obj.setData = function(data) {
+        // Update data
+        if (data) {
+            obj.options.data = data;
+        }
+
+        // Data
+        var data = obj.options.data;
+
+        // Remove content from the DOM
+        container.removeChild(content);
+
+        // Make sure the content container is blank
+        content.innerHTML = '';
+
+        // Reset
+        obj.reset();
+
+        // Reset items
+        obj.items = [];
+
+        // Helpers
+        var items = [];
+        var groups = [];
+
+        // Create elements
+        if (data.length) {
+            // Prepare data
+            for (var i = 0; i < data.length; i++) {
+                // Compatibility
+                if (typeof(data[i]) != 'object') {
+                    // Correct format
+                    obj.options.data[i] = data[i] = { id: data[i], name: data[i] };
+                }
+
+                // Process groups
+                if (data[i].group) {
+                    if (! groups[data[i].group]) {
+                        groups[data[i].group] = [];
+                    }
+                    groups[data[i].group].push(i);
+                } else {
+                    items.push(i);
+                }
+            }
+
+            // Groups
+            var groupNames = Object.keys(groups);
+
+            // Append groups in case exists
+            if (groupNames.length > 0) {
+                for (var i = 0; i < groupNames.length; i++) {
+                    // Group container
+                    var group = document.createElement('div');
+                    group.className = 'jdropdown-group';
+                    // Group name
+                    var groupName = document.createElement('div');
+                    groupName.className = 'jdropdown-group-name';
+                    groupName.innerHTML = groupNames[i];
+                    // Group arrow
+                    var groupArrow = document.createElement('i');
+                    groupArrow.className = 'jdropdown-group-arrow jdropdown-group-arrow-down';
+                    groupName.appendChild(groupArrow);
+                    // Group items
+                    var groupContent = document.createElement('div');
+                    groupContent.className = 'jdropdown-group-items';
+                    for (var j = 0; j < groups[groupNames[i]].length; j++) {
+                        var item = obj.createItem(data[groups[groupNames[i]][j]]);
+                        groupContent.appendChild(item.element);
+                        // Items
+                        obj.items.push(item);
+                    }
+                    // Group itens
+                    group.appendChild(groupName);
+                    group.appendChild(groupArrow);
+                    group.appendChild(groupContent);
+                    content.appendChild(group);
+                }
+            }
+
+            if (items.length) {
+                for (var i = 0; i < items.length; i++) {
+                    var item = obj.createItem(data[items[i]]);
+                    obj.items.push(item);
+                    content.appendChild(item.element);
+                }
+            }
+
+            // Create the Indexes
+            for (var i = 0; i < obj.items.length; i++) {
+                obj.items[i].element.setAttribute('data-index', i);
+            }
+        }
+
+        // Re-insert the content to the container
+        container.appendChild(content);
+    }
+
+    obj.getText = function(asArray) {
+        // Result
+        var result = [];
+        // Append options
+        for (var i = 0; i < obj.selected.length; i++) {
+            if (obj.items[obj.selected[i]]) {
+                result.push(obj.items[obj.selected[i]].text);
+            }
+        }
+
+        if (asArray) {
+            return result;
+        } else {
+            return result.join('; ');
+        }
+    }
+
+    obj.getValue = function(asArray) {
+        // Result
+        var result = [];
+        // Append options
+        for (var i = 0; i < obj.selected.length; i++) {
+            if (obj.items[obj.selected[i]]) {
+                result.push(obj.items[obj.selected[i]].value);
+            }
+        }
+
+        if (asArray) {
+            return result;
+        } else {
+            return result.join(';');
+        }
+    }
+
+    obj.setValue = function(value) {
+        // Remove values
+        for (var i = 0; i < obj.selected.length; i++) {
+            obj.items[obj.selected[i]].element.classList.remove('jdropdown-selected')
+        } 
+
+        // Reset selected
+        obj.selected = [];
+
+        // Set values
+        if (value != null) {
+            if (Array.isArray(value)) {
+                for (var i = 0; i < obj.items.length; i++) {
+                    for (var j = 0; j < value.length; j++) {
+                        if (obj.items[i].value == value[j]) {
+                            // Keep index of the selected item
+                            obj.selected.push(i);
+                            // Visual selection
+                            obj.items[i].element.classList.add('jdropdown-selected');
+                        }
+                    }
+                }
+            } else {
+                for (var i = 0; i < obj.items.length; i++) {
+                    if (obj.items[i].value == value) {
+                        // Keep index of the selected item
+                        obj.selected.push(i);
+                        // Visual selection
+                        obj.items[i].element.classList.add('jdropdown-selected');
+                    }
+                }
+            }
+        }
+
+        // Update labels
+        obj.updateLabel();
+    }
+
+    obj.selectIndex = function(index) {
+        // Only select those existing elements
+        if (obj.items && obj.items[index]) {
+            var index = index = parseInt(index);
+            // Current selection
+            var oldValue = obj.getValue();
+            var oldLabel = obj.getText();
+
+            // Remove cursor style
+            if (obj.currentIndex != null) {
+                obj.items[obj.currentIndex].element.classList.remove('jdropdown-cursor');
+            }
+            // Set cursor style
+            obj.items[index].element.classList.add('jdropdown-cursor');
+
+            // Update cursor position
+            obj.currentIndex = index;
+
+            // Focus behaviour
+            if (! obj.options.multiple) {
+                // Unselect option
+                if (obj.items[index].element.classList.contains('jdropdown-selected')) {
+                    // Reset selected
+                    obj.resetSelected();
+                } else {
+                    // Reset selected
+                    obj.resetSelected();
+                    // Update selected item
+                    obj.items[index].element.classList.add('jdropdown-selected');
+                    // Add to the selected list
+                    obj.selected.push(index);
+                    // Close
+                    obj.close();
+                }
+            } else {
+                // Toggle option
+                if (obj.items[index].element.classList.contains('jdropdown-selected')) {
+                    obj.items[index].element.classList.remove('jdropdown-selected');
+                    // Remove from selected list
+                    var indexToRemove = obj.selected.indexOf(index);
+                    // Remove select
+                    obj.selected.splice(indexToRemove, 1);
+                } else {
+                    // Select element
+                    obj.items[index].element.classList.add('jdropdown-selected');
+                    // Add to the selected list
+                    obj.selected.push(index);
+                }
+
+                // Update labels for multiple dropdown
+                if (! obj.options.autocomplete) {
+                    obj.updateLabel();
+                }
+            }
+
+            // Current selection
+            var newValue = obj.getValue();
+            var newLabel = obj.getText();
+
+            // Events
+            if (typeof(obj.options.onchange) == 'function') {
+                obj.options.onchange(el, index, oldValue, newValue, oldLabel, newLabel);
+            }
+        }
+    }
+
+    obj.selectItem = function(item) {
+        if (jSuites.dropdown.current) {
+            var index = item.getAttribute('data-index');
+            if (index != null) {
+                obj.selectIndex(index);
+            }
+        }
+    }
+
+    obj.find = function(str) {
+        // Force lowercase
+        var str = str ? str.toLowerCase() : null;
+
+        // Append options
+        for (var i = 0; i < obj.items.length; i++) {
+            if (str == null || obj.items[i].textLowerCase.indexOf(str) != -1) {
+                obj.items[i].element.style.display = '';
+            } else {
+                if (obj.selected.indexOf(i) == -1) {
+                    obj.items[i].element.style.display = 'none';
+                } else {
+                    obj.items[i].element.style.display = '';
+                }
+            }
+        }
+
+        var numVisibleItems = function(items) {
+            var visible = 0;
+            for (var j = 0; j < items.length; j++) {
+                if (items[j].style.display != 'none') {
+                    visible++;
+                }
+            }
+            return visible;
+        }
+
+        // Hide groups
+        /*for (var i = 0; i < obj.groups.length; i++) {
+            if (numVisibleItems(obj.groups[i].querySelectorAll('.jdropdown-item'))) {
+                obj.groups[i].children[0].style.display = '';
+            } else {
+                obj.groups[i].children[0].style.display = 'none';
+            }
+        }*/
+    }
+
+    obj.updateLabel = function() {
+        // Update label
+        header.value = obj.getText();
+    }
+
+    obj.open = function() {
+        if (jSuites.dropdown.current != el) {
+            if (jSuites.dropdown.current) {
+                jSuites.dropdown.current.dropdown.close();
+            }
+            jSuites.dropdown.current = el;
+        }
+
+        // Focus
+        if (! el.classList.contains('jdropdown-focus')) {
+            // Add focus
+            el.classList.add('jdropdown-focus');
+
+            // Animation
+            if (jSuites.getWindowWidth() < 800) {
+                if (obj.options.type == null || obj.options.type == 'picker') {
+                    jSuites.slideBottom(container, 1);
+                }
+            }
+
+            // Filter
+            if (obj.options.autocomplete == true) {
+                // Redo search
+                obj.find();
+                // Clear search field
+                header.value = '';
+                header.focus();
+            }
+
+            // Set cursor for the first or first selected element
+            var cursor = (obj.selected && obj.selected[0]) ? obj.selected[0] : 0;
+            obj.updateCursor(cursor);
+
+            // Container Size
+            if (! obj.options.type || obj.options.type == 'default') {
+                const rect = el.getBoundingClientRect();
+                const rectContainer = container.getBoundingClientRect();
+
+                if (obj.options.position) {
+                    container.style.position = 'fixed';
+                    if (window.innerHeight < rect.bottom + rectContainer.height) {
+                        container.style.top = '';
+                        container.style.bottom = (window.innerHeight - rect.top ) + 1 + 'px';
+                    } else {
+                        container.style.top = rect.bottom + 'px';
+                        container.style.bottom = '';
+                    }
+                    container.style.left = rect.left + 'px';
+                } else {
+                    if (window.innerHeight < rect.bottom + rectContainer.height) {
+                        container.style.top = '';
+                        container.style.bottom = rect.height + 1 + 'px';
+                    } else {
+                        container.style.top = '';
+                        container.style.bottom = '';
+                    }
+                }
+
+                container.style.minWidth = rect.width + 'px';
+            }
+        }
+
+        // Events
+        if (typeof(obj.options.onopen) == 'function') {
+            obj.options.onopen(el);
+        }
+    }
+
+    obj.close = function(ignoreEvents) {
+        if (jSuites.dropdown.current) {
+            // Remove controller
+            jSuites.dropdown.current = null
+            // Remove cursor
+            obj.resetCursor();
+            // Update labels
+            obj.updateLabel();
+            // Events
+            if (! ignoreEvents && typeof(obj.options.onclose) == 'function') {
+                obj.options.onclose(el);
+            }
+            // Blur
+            if (header.blur) {
+                header.blur();
+            }
+            // Remove focus
+            el.classList.remove('jdropdown-focus');
+        }
+
+        return obj.getValue();
+    }
+
+    /**
+     * Update position cursor
+     */
+    obj.updateCursor = function(index) {
+        // Set new cursor
+        if (obj.items && obj.items[index] && obj.items[index].element) {
+            // Reset cursor
+            obj.resetCursor();
+
+            // Set new cursor
+            obj.items[index].element.classList.add('jdropdown-cursor');
+
+            // Update position
+            obj.currentIndex = parseInt(index);
+    
+            // Update scroll to the cursor element
+            var container = content.scrollTop;
+            var element = obj.items[obj.currentIndex].element;
+            content.scrollTop = element.offsetTop - element.scrollTop + element.clientTop - 95;
+        }
+    }
+
+    /**
+     * Reset cursor
+     */
+    obj.resetCursor = function() {
+        // Remove current cursor
+        if (obj.currentIndex != null) {
+            // Remove visual cursor
+            if (obj.items && obj.items[obj.currentIndex]) {
+                obj.items[obj.currentIndex].element.classList.remove('jdropdown-cursor');
+            }
+            // Reset cursor
+            obj.currentIndex = null;
+        }
+    }
+
+    /**
+     * Reset cursor
+     */
+    obj.resetSelected = function() {
+        // Unselected all
+        if (obj.selected) {
+            // Remove visual selection
+            for (var i = 0; i < obj.selected.length; i++) {
+                if (obj.items[obj.selected[i]]) {
+                    obj.items[obj.selected[i]].element.classList.remove('jdropdown-selected');
+                }
+            }
+            // Reset current selected items
+            obj.selected = [];
+        }
+    }
+
+    /**
+     * Reset cursor and selected items
+     */
+    obj.reset = function() {
+        // Reset cursor
+        obj.resetCursor();
+
+        // Reset selected
+        obj.resetSelected();
+
+        // Update labels
+        obj.updateLabel();
+    }
+
+    /**
+     * First visible item
+     */
+    obj.firstVisible = function() {
+        var newIndex = null;
+        for (var i = 0; i < obj.items.length; i++) {
+            if (obj.items[i].element.style.display != 'none') {
+                newIndex = i;
+                break;
+            }
+        }
+
+        if (newIndex == null) {
+            return false;
+        }
+
+        obj.updateCursor(newIndex);
+    }
+
+    /**
+     * Navigation
+     */
+    obj.first = function() {
+        var newIndex = null;
+        for (var i = obj.currentIndex - 1; i >= 0; i--) {
+            if (obj.items[i].element.style.display != 'none') {
+                newIndex = i;
+            }
+        }
+
+        if (newIndex == null) {
+            return false;
+        }
+
+        obj.updateCursor(newIndex);
+    }
+
+    obj.last = function() {
+        var newIndex = null;
+        for (var i = obj.currentIndex + 1; i < obj.items.length; i++) {
+            if (obj.items[i].element.style.display != 'none') {
+                newIndex = i;
+            }
+        }
+
+        if (newIndex == null) {
+            return false;
+        }
+
+        obj.updateCursor(newIndex);
+    }
+
+    obj.next = function() {
+        var newIndex = null;
+        for (var i = obj.currentIndex + 1; i < obj.items.length; i++) {
+            if (obj.items[i].element.style.display != 'none') {
+                newIndex = i;
+                break;
+            }
+        }
+
+        if (newIndex == null) {
+            return false;
+        }
+
+        obj.updateCursor(newIndex);
+    }
+
+    obj.prev = function() {
+        var newIndex = null;
+        for (var i = obj.currentIndex - 1; i >= 0; i--) {
+            if (obj.items[i].element.style.display != 'none') {
+                newIndex = i;
+                break;
+            }
+        }
+
+        if (newIndex == null) {
+            return false;
+        }
+
+        obj.updateCursor(newIndex);
+    }
+
+    if (! jSuites.dropdown.hasEvents) {
+        if ('ontouchsend' in document.documentElement === true) {
+            document.addEventListener('touchsend', jSuites.dropdown.mouseup);
+        } else {
+            document.addEventListener('mouseup', jSuites.dropdown.mouseup);
+        }
+        document.addEventListener('keydown', jSuites.dropdown.onkeydown);
+
+        jSuites.dropdown.hasEvents = true;
+    }
+
+    // Start dropdown
+    obj.init();
+
+    // Keep object available from the node
+    el.dropdown = obj;
+
+    return obj;
+});
+
+jSuites.dropdown.hasEvents = false;
+
+jSuites.dropdown.mouseup = function(e) {
+    var element = jSuites.getElement(e.target, 'jdropdown');
+    if (element) {
+        var dropdown = element.dropdown;
+        if (e.target.classList.contains('jdropdown-header')) {
+            if (element.classList.contains('jdropdown-focus') && element.classList.contains('jdropdown-default')) {
+                dropdown.close();
+            } else {
+                dropdown.open();
+            }
+        } else if (e.target.classList.contains('jdropdown-group-name')) {
+            var items = e.target.nextSibling.children;
+            if (e.target.nextSibling.style.display != 'none') {
+                for (var i = 0; i < items.length; i++) {
+                    if (items[i].style.display != 'none') {
+                        dropdown.selectItem(items[i]);
+                    }
+                }
+            }
+        } else if (e.target.classList.contains('jdropdown-group-arrow')) {
+            if (e.target.classList.contains('jdropdown-group-arrow-down')) {
+                e.target.classList.remove('jdropdown-group-arrow-down');
+                e.target.classList.add('jdropdown-group-arrow-up');
+                e.target.parentNode.nextSibling.style.display = 'none';
+            } else {
+                e.target.classList.remove('jdropdown-group-arrow-up');
+                e.target.classList.add('jdropdown-group-arrow-down');
+                e.target.parentNode.nextSibling.style.display = '';
+            }
+        } else if (e.target.classList.contains('jdropdown-item')) {
+            dropdown.selectItem(e.target);
+        } else if (e.target.classList.contains('jdropdown-image')) {
+            dropdown.selectIndex(e.target.parentNode.getAttribute('data-index'));
+        } else if (e.target.classList.contains('jdropdown-description')) {
+            dropdown.selectIndex(e.target.parentNode.getAttribute('data-index'));
+        } else if (e.target.classList.contains('jdropdown-title')) {
+            dropdown.selectIndex(e.target.parentNode.parentNode.getAttribute('data-index'));
+        } else if (e.target.classList.contains('jdropdown-close') || e.target.classList.contains('jdropdown-backdrop')) {
+            // Close
+            dropdown.close();
+        }
+
+        e.stopPropagation();
+        e.preventDefault();
+    } else {
+        if (jSuites.dropdown.current) {
+            jSuites.dropdown.current.dropdown.close();
+        }
+    }
+}
+
+
+// Keydown controls
+jSuites.dropdown.onkeydown = function(e) {
+    if (jSuites.dropdown.current) {
+        // Element
+        var element = jSuites.dropdown.current.dropdown;
+        // Index
+        var index = element.currentIndex;
+
+        if (e.shiftKey) {
+
+        } else {
+            if (e.which == 13 || e.which == 27 || e.which == 35 || e.which == 36 || e.which == 38 || e.which == 40) {
+                // Move cursor
+                if (e.which == 13) {
+                    element.selectIndex(index)
+                } else if (e.which == 38) {
+                    if (index == null) {
+                        element.firstVisible();
+                    } else if (index > 0) {
+                        element.prev();
+                    }
+                } else if (e.which == 40) {
+                    if (index == null) {
+                        element.firstVisible();
+                    } else if (index + 1 < element.options.data.length) {
+                        element.next();
+                    }
+                } else if (e.which == 36) {
+                    element.first();
+                } else if (e.which == 35) {
+                    element.last();
+                } else if (e.which == 27) {
+                    element.close();
+                }
+
+                e.stopPropagation();
+                e.preventDefault();
+            }
+        }
+    }
+}
+
+jSuites.dropdown.extractFromDom = function(el, options) {
+    // Keep reference
+    var select = el;
+    if (! options) {
+        options = {};
+    }
+    // Prepare configuration
+    if (el.getAttribute('multiple') && (! options || options.multiple == undefined)) {
+        options.multiple = true;
+    }
+    if (el.getAttribute('placeholder') && (! options || options.placeholder == undefined)) {
+        options.placeholder = el.getAttribute('placeholder');
+    }
+    if (el.getAttribute('data-autocomplete') && (! options || options.autocomplete == undefined)) {
+        options.autocomplete = true;
+    }
+    if (! options || options.width == undefined) {
+        options.width = el.offsetWidth;
+    }
+    if (el.value && (! options || options.value == undefined)) {
+        options.value = el.value;
+    }
+    if (! options || options.data == undefined) {
+        options.data = [];
+        for (var j = 0; j < el.children.length; j++) {
+            if (el.children[j].tagName == 'OPTGROUP') {
+                for (var i = 0; i < el.children[j].children.length; i++) {
+                    options.data.push({
+                        id: el.children[j].children[i].value,
+                        name: el.children[j].children[i].innerHTML,
+                        group: el.children[j].getAttribute('label'),
+                    });
+                }
+            } else {
+                options.data.push({
+                    id: el.children[j].value,
+                    name: el.children[j].innerHTML,
+                });
+            }
+        }
+    }
+    if (! options || options.onchange == undefined) {
+        options.onchange = function(a,b,c,d) {
+            if (options.multiple == true) {
+                if (obj.items[b].classList.contains('jdropdown-selected')) {
+                    select.options[b].setAttribute('selected', 'selected');
+                } else {
+                    select.options[b].removeAttribute('selected');
+                }
+            } else {
+                select.value = d;
+            }
+        }
+    }
+    // Create DIV
+    var div = document.createElement('div');
+    el.parentNode.insertBefore(div, el);
+    el.style.display = 'none';
+    el = div;
+
+    return { el:el, options:options };
+}
+
+/**
+ * (c) jTools Text Editor
+ * https://github.com/paulhodel/jtools
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Inline richtext editor
+ */
+
+jSuites.editor = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // Default configuration
+    var defaults = {
+        // Initial HTML content
+        value: null,
+        // Initial snippet
+        snippet: null,
+        // Add toolbar
+        toolbar: null,
+        // Max height
+        maxHeight: null,
+        // Website parser is to read websites and images from cross domain
+        remoteParser: null,
+        // Key from youtube to read properties from URL
+        youtubeKey: null,
+        // User list
+        userSearch: null,
+        // Parse URL
+        parseURL: false,
+        // Accept drop files
+        dropZone: true,
+        dropAsAttachment: false,
+        acceptImages: true,
+        acceptFiles: false,
+        maxFileSize: 5000000, 
+        // Border
+        border: true,
+        padding: true,
+        focus: false,
+        // Events
+        onclick: null,
+        onfocus: null,
+        onblur: null,
+        onload: null,
+        onenter: null,
+        onkeyup: null,
+        onkeydown: null,
+    };
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Private controllers
+    var imageResize = 0;
+    var editorTimer = null;
+    var editorAction = null;
+
+    // Make sure element is empty
+    el.innerHTML = '';
+
+    // Prepare container
+    el.classList.add('jeditor-container');
+
+    // Padding
+    if (obj.options.padding == true) {
+        el.classList.add('jeditor-padding');
+    }
+
+    // Border
+    if (obj.options.border == false) {
+        el.style.border = '0px';
+    }
+
+    // Snippet
+    var snippet = document.createElement('div');
+    snippet.className = 'snippet';
+    snippet.setAttribute('contenteditable', false);
+
+    // Toolbar
+    var toolbar = document.createElement('div');
+    toolbar.className = 'jeditor-toolbar';
+
+    // Create editor
+    var editor = document.createElement('div');
+    editor.setAttribute('contenteditable', true);
+    editor.setAttribute('spellcheck', false);
+    editor.className = 'jeditor';
+
+    // Max height
+    if (obj.options.maxHeight) {
+        editor.style.overflowY = 'auto';
+        editor.style.maxHeight = obj.options.maxHeight;
+    }
+
+    // Set editor initial value
+    if (obj.options.value) {
+        var value = obj.options.value;
+    } else {
+        var value = el.innerHTML ? el.innerHTML : ''; 
+    }
+
+    if (! value) {
+        var value = '<br>';
+    }
+
+    /**
+     * Extract images from a HTML string
+     */
+    var extractImageFromHtml = function(html) {
+        // Create temp element
+        var div = document.createElement('div');
+        div.innerHTML = html;
+
+        // Extract images
+        var img = div.querySelectorAll('img');
+
+        if (img.length) {
+            for (var i = 0; i < img.length; i++) {
+                obj.addImage(img[i].src);
+            }
+        }
+    }
+
+    /**
+     * Insert node at caret
+     */
+    var insertNodeAtCaret = function(newNode) {
+        var sel, range;
+
+        if (window.getSelection) {
+            sel = window.getSelection();
+            if (sel.rangeCount) {
+                range = sel.getRangeAt(0);
+                var selectedText = range.toString();
+                range.deleteContents();
+                range.insertNode(newNode); 
+                // move the cursor after element
+                range.setStartAfter(newNode);
+                range.setEndAfter(newNode); 
+                sel.removeAllRanges();
+                sel.addRange(range);
+            }
+        }
+    }
+
+    /**
+     * Append snippet or thumbs in the editor
+     * @Param object data
+     */
+    var appendElement = function(data) {
+        // Reset snippet
+        snippet.innerHTML = '';
+
+        if (data.image) {
+            var div = document.createElement('div');
+            div.className = 'snippet-image';
+            div.setAttribute('data-k', 'image');
+            snippet.appendChild(div);
+
+            var image = document.createElement('img');
+            image.src = data.image;
+            div.appendChild(image);
+        }
+
+        var div = document.createElement('div');
+        div.className = 'snippet-title';
+        div.setAttribute('data-k', 'title');
+        div.innerHTML = data.title;
+        snippet.appendChild(div);
+
+        var div = document.createElement('div');
+        div.className = 'snippet-description';
+        div.setAttribute('data-k', 'description');
+        div.innerHTML = data.description;
+        snippet.appendChild(div);
+
+        var div = document.createElement('div');
+        div.className = 'snippet-host';
+        div.setAttribute('data-k', 'host');
+        div.innerHTML = data.host;
+        snippet.appendChild(div);
+
+        var div = document.createElement('div');
+        div.className = 'snippet-url';
+        div.setAttribute('data-k', 'url');
+        div.innerHTML = data.url;
+        snippet.appendChild(div);
+
+        editor.appendChild(snippet);
+    }
+
+    var verifyEditor = function() {
+        clearTimeout(editorTimer);
+        editorTimer = setTimeout(function() {
+            var snippet = editor.querySelector('.snippet');
+            var thumbsContainer = el.querySelector('.jeditor-thumbs-container');
+
+            if (! snippet && ! thumbsContainer) {
+                var html = editor.innerHTML.replace(/\n/g, ' ');
+                var container = document.createElement('div');
+                container.innerHTML = html;
+                var thumbsContainer = container.querySelector('.jeditor-thumbs-container');
+                if (thumbsContainer) {
+                    thumbsContainer.remove();
+                }
+                var text = container.innerText; 
+                var url = jSuites.editor.detectUrl(text);
+
+                if (url) {
+                    if (url[0].substr(-3) == 'jpg' || url[0].substr(-3) == 'png' || url[0].substr(-3) == 'gif') {
+                        if (jSuites.editor.getDomain(url[0]) == window.location.hostname) {
+                            obj.importImage(url[0], '');
+                        } else {
+                            obj.importImage(obj.options.remoteParser + url[0], '');
+                        }
+                    } else {
+                        var id = jSuites.editor.youtubeParser(url[0]);
+
+                        if (id) {
+                            obj.getYoutube(id);
+                        } else {
+                            obj.getWebsite(url[0]);
+                        }
+                    }
+                }
+            }
+        }, 1000);
+    }
+
+    obj.parseContent = function() {
+        verifyEditor();
+    }
+
+    /**
+     * Get metadata from a youtube video
+     */
+    obj.getYoutube = function(id) {
+        if (! obj.options.youtubeKey) {
+            console.error('The youtubeKey is not defined');
+        } else {
+            jSuites.ajax({
+                url: 'https://www.googleapis.com/youtube/v3/videos?part=snippet,statistics&key=' + obj.options.youtubeKey + '&id=' + id,
+                method: 'GET',
+                dataType: 'json',
+                success: function(result) {
+                    // Only valid elements to be appended
+                    if (result.items && result.items[0]) {
+                        var p = {
+                            title: '',
+                            description: '',
+                            image: '',
+                            host: 'www.youtube.com',
+                            url: 'https://www.youtube.com?watch=' + id,
+                        }
+                        if (result.items[0].snippet.title) {
+                            p.title = result.items[0].snippet.title;
+                        }
+                        if (result.items[0].snippet.description) {
+                            p.description = result.items[0].snippet.description;
+
+                            if (p.description.length > 150) {
+                                p.description = p.description.substr(0, 150) + '...';
+                            }
+                        }
+                        if (result.items[0].snippet.thumbnails.medium.url) {
+                            p.image = result.items[0].snippet.thumbnails.medium.url;
+                        }
+
+                        appendElement(p);
+                    }
+                }
+            });
+        }
+    }
+
+    /**
+     * Get meta information from a website
+     */
+    obj.getWebsite = function(url) {
+        if (! obj.options.remoteParser) {
+            console.log('The remoteParser is not defined');
+        } else {
+            jSuites.ajax({
+                url: obj.options.remoteParser + encodeURI(url.trim()),
+                method: 'GET',
+                dataType: 'json',
+                success: function(result) {
+                    var p = {
+                        title: '',
+                        description: '',
+                        image: '',
+                        host: url,
+                        url: url,
+                    }
+
+                    if (result.title) {
+                        p.title = result.title;
+                    }
+                    if (result.description) {
+                        p.description = result.description;
+                    }
+                    if (result.image) {
+                        p.image = result.image;
+                    } else if (result['og:image']) {
+                        p.image = result['og:image'];
+                    }
+                    if (result.host) {
+                        p.host = result.host;
+                    }
+                    if (result.url) {
+                        p.url = result.url;
+                    }
+
+                    appendElement(p);
+                }
+            });
+        }
+    }
+
+    /**
+     * Set editor value
+     */
+    obj.setData = function(html) {
+        editor.innerHTML = html;
+        jSuites.editor.setCursor(editor, true);
+    }
+
+    /**
+     * Get editor data
+     */
+    obj.getData = function(json) {
+        if (! json) {
+            var data = editor.innerHTML;
+        } else {
+            var data = {
+                content : '',
+            }
+
+            // Get tag users
+            var tagged = editor.querySelectorAll('.post-tag');
+            if (tagged.length) {
+                data.users = [];
+                for (var i = 0; i < tagged.length; i++) {
+                    var userId = tagged[i].getAttribute('data-user');
+                    if (userId) {
+                        data.users.push(userId);
+                    }
+                }
+                data.users = data.users.join(',');
+            }
+
+            if (snippet.innerHTML) {
+                var index = 0;
+                data.snippet = {};
+                for (var i = 0; i < snippet.children.length; i++) {
+                    // Get key from element
+                    var key = snippet.children[i].getAttribute('data-k');
+                    if (key) {
+                        if (key == 'image') {
+                            data.snippet.image = snippet.children[i].children[0].getAttribute('src');
+                        } else {
+                            data.snippet[key] = snippet.children[i].innerHTML;
+                        }
+                    }
+                }
+
+                snippet.innerHTML = '';
+                snippet.remove();
+            }
+
+            var text = editor.innerHTML;
+            text = text.replace(/<br>/g, "\n");
+            text = text.replace(/<\/div>/g, "<\/div>\n");
+            text = text.replace(/<(?:.|\n)*?>/gm, "");
+            data.content = text.trim();
+            data = JSON.stringify(data);
+        }
+
+        return data;
+    }
+
+    // Reset
+    obj.reset = function() {
+        editor.innerHTML = '';
+    }
+
+    obj.addPdf = function(data) {
+        if (data.result.substr(0,4) != 'data') {
+            console.error('Invalid source');
+        } else {
+            var canvas = document.createElement('canvas');
+            canvas.width = 60;
+            canvas.height = 60;
+
+            var img = new Image();
+            var ctx = canvas.getContext('2d');
+            ctx.drawImage(img, 0, 0, canvas.width, canvas.height);
+
+            canvas.toBlob(function(blob) {
+                var newImage = document.createElement('img');
+                newImage.src = window.URL.createObjectURL(blob);
+                newImage.setAttribute('data-extension', 'pdf');
+                if (data.name) {
+                    newImage.setAttribute('data-name', data.name);
+                }
+                if (data.size) {
+                    newImage.setAttribute('data-size', data.size);
+                }
+                if (data.date) {
+                    newImage.setAttribute('data-date', data.date);
+                }
+                newImage.className = 'jfile pdf';
+
+                insertNodeAtCaret(newImage);
+                jSuites.files[newImage.src] = data.result.substr(data.result.indexOf(',') + 1);
+            });
+        }
+    }
+
+    obj.addImage = function(src, name, size, date) {
+        if (src.substr(0,4) != 'data' && ! obj.options.remoteParser) {
+            console.error('remoteParser not defined in your initialization');
+        } else {
+            // This is to process cross domain images
+            if (src.substr(0,4) == 'data') {
+                var extension = src.split(';')
+                extension = extension[0].split('/');
+                extension = extension[1];
+            } else {
+                var extension = src.substr(src.lastIndexOf('.') + 1);
+                // Work for cross browsers
+                src = obj.options.remoteParser + src;
+            }
+
+            var img = new Image();
+
+            img.onload = function onload() {
+                var canvas = document.createElement('canvas');
+                canvas.width = img.width;
+                canvas.height = img.height;
+
+                var ctx = canvas.getContext('2d');
+                ctx.drawImage(img, 0, 0, canvas.width, canvas.height);
+
+                canvas.toBlob(function(blob) {
+                    var newImage = document.createElement('img');
+                    newImage.src = window.URL.createObjectURL(blob);
+                    newImage.setAttribute('tabindex', '900');
+                    newImage.setAttribute('data-extension', extension);
+                    if (name) {
+                        newImage.setAttribute('data-name', name);
+                    }
+                    if (size) {
+                        newImage.setAttribute('data-size', size);
+                    }
+                    if (date) {
+                        newImage.setAttribute('data-date', date);
+                    }
+                    newImage.className = 'jfile';
+                    var content = canvas.toDataURL();
+                    insertNodeAtCaret(newImage);
+
+                    jSuites.files[newImage.src] = content.substr(content.indexOf(',') + 1);
+                });
+            };
+
+            img.src = src;
+        }
+    }
+
+    obj.addFile = function(files) {
+        var reader = [];
+
+        for (var i = 0; i < files.length; i++) {
+            if (files[i].size > obj.options.maxFileSize) {
+                alert('The file is too big');
+            } else {
+                // Only PDF or Images
+                var type = files[i].type.split('/');
+
+                if (type[0] == 'image') {
+                    type = 1;
+                } else if (type[1] == 'pdf') {
+                    type = 2;
+                } else {
+                    type = 0;
+                }
+
+                if (type) {
+                    // Create file
+                    reader[i] = new FileReader();
+                    reader[i].index = i;
+                    reader[i].type = type;
+                    reader[i].name = files[i].name;
+                    reader[i].date = files[i].lastModified;
+                    reader[i].size = files[i].size;
+                    reader[i].addEventListener("load", function (data) {
+                        // Get result
+                        if (data.target.type == 2) {
+                            if (obj.options.acceptFiles == true) {
+                                obj.addPdf(data.target);
+                            }
+                        } else {
+                            obj.addImage(data.target.result, data.target.name, data.total, data.target.lastModified);
+                        }
+                    }, false);
+
+                    reader[i].readAsDataURL(files[i])
+                } else {
+                    alert('The extension is not allowed');
+                }
+            }
+        }
+    }
+
+    // Destroy
+    obj.destroy = function() {
+        editor.removeEventListener('mouseup', editorMouseUp);
+        editor.removeEventListener('mousedown', editorMouseDown);
+        editor.removeEventListener('mousemove', editorMouseMove);
+        editor.removeEventListener('keyup', editorKeyUp);
+        editor.removeEventListener('keydown', editorKeyDown);
+        editor.removeEventListener('dragstart', editorDragStart);
+        editor.removeEventListener('dragenter', editorDragEnter);
+        editor.removeEventListener('dragover', editorDragOver);
+        editor.removeEventListener('drop', editorDrop);
+        editor.removeEventListener('paste', editorPaste);
+
+        if (typeof(obj.options.onblur) == 'function') {
+            editor.removeEventListener('blur', editorBlur);
+        }
+        if (typeof(obj.options.onfocus) == 'function') {
+            editor.removeEventListener('focus', editorFocus);
+        }
+
+        el.editor = null;
+        el.classList.remove('jeditor-container');
+
+        toolbar.remove();
+        snippet.remove();
+        editor.remove();
+    }
+
+    var isLetter = function (str) {
+        var regex = /([\u0041-\u005A\u0061-\u007A\u00AA\u00B5\u00BA\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02C1\u02C6-\u02D1\u02E0-\u02E4\u02EC\u02EE\u0370-\u0374\u0376\u0377\u037A-\u037D\u0386\u0388-\u038A\u038C\u038E-\u03A1\u03A3-\u03F5\u03F7-\u0481\u048A-\u0527\u0531-\u0556\u0559\u0561-\u0587\u05D0-\u05EA\u05F0-\u05F2\u0620-\u064A\u066E\u066F\u0671-\u06D3\u06D5\u06E5\u06E6\u06EE\u06EF\u06FA-\u06FC\u06FF\u0710\u0712-\u072F\u074D-\u07A5\u07B1\u07CA-\u07EA\u07F4\u07F5\u07FA\u0800-\u0815\u081A\u0824\u0828\u0840-\u0858\u08A0\u08A2-\u08AC\u0904-\u0939\u093D\u0950\u0958-\u0961\u0971-\u0977\u0979-\u097F\u0985-\u098C\u098F\u0990\u0993-\u09A8\u09AA-\u09B0\u09B2\u09B6-\u09B9\u09BD\u09CE\u09DC\u09DD\u09DF-\u09E1\u09F0\u09F1\u0A05-\u0A0A\u0A0F\u0A10\u0A13-\u0A28\u0A2A-\u0A30\u0A32\u0A33\u0A35\u0A36\u0A38\u0A39\u0A59-\u0A5C\u0A5E\u0A72-\u0A74\u0A85-\u0A8D\u0A8F-\u0A91\u0A93-\u0AA8\u0AAA-\u0AB0\u0AB2\u0AB3\u0AB5-\u0AB9\u0ABD\u0AD0\u0AE0\u0AE1\u0B05-\u0B0C\u0B0F\u0B10\u0B13-\u0B28\u0B2A-\u0B30\u0B32\u0B33\u0B35-\u0B39\u0B3D\u0B5C\u0B5D\u0B5F-\u0B61\u0B71\u0B83\u0B85-\u0B8A\u0B8E-\u0B90\u0B92-\u0B95\u0B99\u0B9A\u0B9C\u0B9E\u0B9F\u0BA3\u0BA4\u0BA8-\u0BAA\u0BAE-\u0BB9\u0BD0\u0C05-\u0C0C\u0C0E-\u0C10\u0C12-\u0C28\u0C2A-\u0C33\u0C35-\u0C39\u0C3D\u0C58\u0C59\u0C60\u0C61\u0C85-\u0C8C\u0C8E-\u0C90\u0C92-\u0CA8\u0CAA-\u0CB3\u0CB5-\u0CB9\u0CBD\u0CDE\u0CE0\u0CE1\u0CF1\u0CF2\u0D05-\u0D0C\u0D0E-\u0D10\u0D12-\u0D3A\u0D3D\u0D4E\u0D60\u0D61\u0D7A-\u0D7F\u0D85-\u0D96\u0D9A-\u0DB1\u0DB3-\u0DBB\u0DBD\u0DC0-\u0DC6\u0E01-\u0E30\u0E32\u0E33\u0E40-\u0E46\u0E81\u0E82\u0E84\u0E87\u0E88\u0E8A\u0E8D\u0E94-\u0E97\u0E99-\u0E9F\u0EA1-\u0EA3\u0EA5\u0EA7\u0EAA\u0EAB\u0EAD-\u0EB0\u0EB2\u0EB3\u0EBD\u0EC0-\u0EC4\u0EC6\u0EDC-\u0EDF\u0F00\u0F40-\u0F47\u0F49-\u0F6C\u0F88-\u0F8C\u1000-\u102A\u103F\u1050-\u1055\u105A-\u105D\u1061\u1065\u1066\u106E-\u1070\u1075-\u1081\u108E\u10A0-\u10C5\u10C7\u10CD\u10D0-\u10FA\u10FC-\u1248\u124A-\u124D\u1250-\u1256\u1258\u125A-\u125D\u1260-\u1288\u128A-\u128D\u1290-\u12B0\u12B2-\u12B5\u12B8-\u12BE\u12C0\u12C2-\u12C5\u12C8-\u12D6\u12D8-\u1310\u1312-\u1315\u1318-\u135A\u1380-\u138F\u13A0-\u13F4\u1401-\u166C\u166F-\u167F\u1681-\u169A\u16A0-\u16EA\u1700-\u170C\u170E-\u1711\u1720-\u1731\u1740-\u1751\u1760-\u176C\u176E-\u1770\u1780-\u17B3\u17D7\u17DC\u1820-\u1877\u1880-\u18A8\u18AA\u18B0-\u18F5\u1900-\u191C\u1950-\u196D\u1970-\u1974\u1980-\u19AB\u19C1-\u19C7\u1A00-\u1A16\u1A20-\u1A54\u1AA7\u1B05-\u1B33\u1B45-\u1B4B\u1B83-\u1BA0\u1BAE\u1BAF\u1BBA-\u1BE5\u1C00-\u1C23\u1C4D-\u1C4F\u1C5A-\u1C7D\u1CE9-\u1CEC\u1CEE-\u1CF1\u1CF5\u1CF6\u1D00-\u1DBF\u1E00-\u1F15\u1F18-\u1F1D\u1F20-\u1F45\u1F48-\u1F4D\u1F50-\u1F57\u1F59\u1F5B\u1F5D\u1F5F-\u1F7D\u1F80-\u1FB4\u1FB6-\u1FBC\u1FBE\u1FC2-\u1FC4\u1FC6-\u1FCC\u1FD0-\u1FD3\u1FD6-\u1FDB\u1FE0-\u1FEC\u1FF2-\u1FF4\u1FF6-\u1FFC\u2071\u207F\u2090-\u209C\u2102\u2107\u210A-\u2113\u2115\u2119-\u211D\u2124\u2126\u2128\u212A-\u212D\u212F-\u2139\u213C-\u213F\u2145-\u2149\u214E\u2183\u2184\u2C00-\u2C2E\u2C30-\u2C5E\u2C60-\u2CE4\u2CEB-\u2CEE\u2CF2\u2CF3\u2D00-\u2D25\u2D27\u2D2D\u2D30-\u2D67\u2D6F\u2D80-\u2D96\u2DA0-\u2DA6\u2DA8-\u2DAE\u2DB0-\u2DB6\u2DB8-\u2DBE\u2DC0-\u2DC6\u2DC8-\u2DCE\u2DD0-\u2DD6\u2DD8-\u2DDE\u2E2F\u3005\u3006\u3031-\u3035\u303B\u303C\u3041-\u3096\u309D-\u309F\u30A1-\u30FA\u30FC-\u30FF\u3105-\u312D\u3131-\u318E\u31A0-\u31BA\u31F0-\u31FF\u3400-\u4DB5\u4E00-\u9FCC\uA000-\uA48C\uA4D0-\uA4FD\uA500-\uA60C\uA610-\uA61F\uA62A\uA62B\uA640-\uA66E\uA67F-\uA697\uA6A0-\uA6E5\uA717-\uA71F\uA722-\uA788\uA78B-\uA78E\uA790-\uA793\uA7A0-\uA7AA\uA7F8-\uA801\uA803-\uA805\uA807-\uA80A\uA80C-\uA822\uA840-\uA873\uA882-\uA8B3\uA8F2-\uA8F7\uA8FB\uA90A-\uA925\uA930-\uA946\uA960-\uA97C\uA984-\uA9B2\uA9CF\uAA00-\uAA28\uAA40-\uAA42\uAA44-\uAA4B\uAA60-\uAA76\uAA7A\uAA80-\uAAAF\uAAB1\uAAB5\uAAB6\uAAB9-\uAABD\uAAC0\uAAC2\uAADB-\uAADD\uAAE0-\uAAEA\uAAF2-\uAAF4\uAB01-\uAB06\uAB09-\uAB0E\uAB11-\uAB16\uAB20-\uAB26\uAB28-\uAB2E\uABC0-\uABE2\uAC00-\uD7A3\uD7B0-\uD7C6\uD7CB-\uD7FB\uF900-\uFA6D\uFA70-\uFAD9\uFB00-\uFB06\uFB13-\uFB17\uFB1D\uFB1F-\uFB28\uFB2A-\uFB36\uFB38-\uFB3C\uFB3E\uFB40\uFB41\uFB43\uFB44\uFB46-\uFBB1\uFBD3-\uFD3D\uFD50-\uFD8F\uFD92-\uFDC7\uFDF0-\uFDFB\uFE70-\uFE74\uFE76-\uFEFC\uFF21-\uFF3A\uFF41-\uFF5A\uFF66-\uFFBE\uFFC2-\uFFC7\uFFCA-\uFFCF\uFFD2-\uFFD7\uFFDA-\uFFDC]+)/g;
+        return str.match(regex) ? 1 : 0;
+    }
+
+    // Event handlers
+    var editorMouseUp = function(e) {
+        editorAction = false;
+    }
+
+    var editorMouseDown = function(e) {
+        var close = function(snippet) {
+            var rect = snippet.getBoundingClientRect();
+            if (rect.width - (e.clientX - rect.left) < 40 && e.clientY - rect.top < 40) {
+                snippet.innerHTML = '';
+                snippet.remove();
+            }
+        }
+
+        if (e.target.tagName == 'IMG') {
+            if (e.target.style.cursor) {
+                var rect = e.target.getBoundingClientRect();
+                editorAction = {
+                    e: e.target,
+                    x: e.clientX,
+                    y: e.clientY,
+                    w: rect.width,
+                    h: rect.height,
+                    d: e.target.style.cursor,
+                }
+
+                if (! e.target.style.width) {
+                    e.target.style.width = rect.width + 'px';
+                }
+
+                if (! e.target.style.height) {
+                    e.target.style.height = rect.height + 'px';
+                }
+
+                var s = window.getSelection();
+                if (s.rangeCount) {
+                    for (var i = 0; i < s.rangeCount; i++) {
+                        s.removeRange(s.getRangeAt(i));
+                    }
+                }
+            } else {
+                editorAction = true;
+            }
+        } else { 
+            if (e.target.classList.contains('snippet')) {
+                close(e.target);
+            } else if (e.target.parentNode.classList.contains('snippet')) {
+                close(e.target.parentNode);
+            }
+
+            editorAction = true;
+        }
+    }
+
+    var editorMouseMove = function(e) {
+        if (e.target.tagName == 'IMG') {
+            if (e.target.getAttribute('tabindex')) {
+                var rect = e.target.getBoundingClientRect();
+                if (e.clientY - rect.top < 5) {
+                    if (rect.width - (e.clientX - rect.left) < 5) {
+                        e.target.style.cursor = 'ne-resize';
+                    } else if (e.clientX - rect.left < 5) {
+                        e.target.style.cursor = 'nw-resize';
+                    } else {
+                        e.target.style.cursor = 'n-resize';
+                    }
+                } else if (rect.height - (e.clientY - rect.top) < 5) {
+                    if (rect.width - (e.clientX - rect.left) < 5) {
+                        e.target.style.cursor = 'se-resize';
+                    } else if (e.clientX - rect.left < 5) {
+                        e.target.style.cursor = 'sw-resize';
+                    } else {
+                        e.target.style.cursor = 's-resize';
+                    }
+                } else if (rect.width - (e.clientX - rect.left) < 5) {
+                    e.target.style.cursor = 'e-resize';
+                } else if (e.clientX - rect.left < 5) {
+                    e.target.style.cursor = 'w-resize';
+                } else {
+                    e.target.style.cursor = '';
+                }
+            }
+        }
+
+        // Move
+        if (e.which == 1 && editorAction && editorAction.d) {
+            if (editorAction.d == 'e-resize' || editorAction.d == 'ne-resize' ||  editorAction.d == 'se-resize') {
+                editorAction.e.style.width = (editorAction.w + (e.clientX - editorAction.x)) + 'px';
+
+                if (e.shiftKey) {
+                    var newHeight = (e.clientX - editorAction.x) * (editorAction.h / editorAction.w);
+                    editorAction.e.style.height = editorAction.h + newHeight + 'px';
+                } else {
+                    var newHeight =  null;
+                }
+            }
+
+            if (! newHeight) {
+                if (editorAction.d == 's-resize' || editorAction.d == 'se-resize' || editorAction.d == 'sw-resize') {
+                    if (! e.shiftKey) {
+                        editorAction.e.style.height = editorAction.h + (e.clientY - editorAction.y);
+                    }
+                }
+            }
+        }
+    }
+
+    var editorKeyUp = function(e) {
+        if (! editor.innerHTML) {
+            editor.innerHTML = '<div><br></div>';
+        }
+
+        if (typeof(obj.options.onkeyup) == 'function') { 
+            obj.options.onkeyup(e, el);
+        }
+    }
+
+
+    var editorKeyDown = function(e) {
+        // Check for URL
+        if (obj.options.parseURL == true) {
+            verifyEditor();
+        }
+
+        // Closable
+        if (typeof(obj.options.onenter) == 'function' && e.which == 13) {
+            var data = obj.getData();
+            obj.options.onenter(obj, el, data, e);
+        }
+
+        if (typeof(obj.options.onkeydown) == 'function') { 
+            obj.options.onkeydown(e, el);
+        }
+    }
+
+    var editorPaste = function(e) {
+        if (e.clipboardData || e.originalEvent.clipboardData) {
+            var html = (e.originalEvent || e).clipboardData.getData('text/html');
+            var text = (e.originalEvent || e).clipboardData.getData('text/plain');
+            var file = (e.originalEvent || e).clipboardData.files
+        } else if (window.clipboardData) {
+            var html = window.clipboardData.getData('Html');
+            var text = window.clipboardData.getData('Text');
+            var file = window.clipboardData.files
+        }
+
+        if (file.length) {
+            // Paste a image from the clipboard
+            obj.addFile(file);
+        } else {
+            // Paste text
+            text = text.split('\r\n');
+            var str = '';
+            if (e.target.nodeName == 'DIV' && e.target.classList.contains('jeditor')) {
+                for (var i = 0; i < text.length; i++) {
+                    var tmp = document.createElement('div');
+                    if (text[i]) {
+                        tmp.innerHTML = text[i];
+                    } else {
+                        tmp.innerHTML = '<br/>';
+                    }
+                    e.target.appendChild(tmp);
+                }
+            } else {
+                for (var i = 0; i < text.length; i++) {
+                    if (text[i]) {
+                        str += '<div>' + text[i] + "</div>\r\n";
+                    }
+                }
+                // Insert text
+                document.execCommand('insertHtml', false, str);
+            }
+
+            // Extra images from the paste
+            if (obj.options.acceptImages == true) {
+                extractImageFromHtml(html);
+            }
+        }
+
+        e.preventDefault();
+    }
+
+    var editorDragStart = function(e) {
+        if (editorAction && editorAction.e) {
+            e.preventDefault();
+        }
+    }
+
+    var editorDragEnter = function(e) {
+        if (editorAction || obj.options.dropZone == false) {
+            // Do nothing
+        } else {
+            el.classList.add('jeditor-dragging');
+        }
+    }
+
+    var editorDragOver = function(e) {
+        if (editorAction || obj.options.dropZone == false) {
+            // Do nothing
+        } else {
+            if (editorTimer) {
+                clearTimeout(editorTimer);
+            }
+
+            editorTimer = setTimeout(function() {
+                el.classList.remove('jeditor-dragging');
+            }, 100);
+        }
+    }
+
+    var editorDrop = function(e) {
+        if (editorAction || obj.options.dropZone == false) {
+            // Do nothing
+        } else {
+            // Position caret on the drop
+            var range = null;
+            if (document.caretRangeFromPoint) {
+                range=document.caretRangeFromPoint(e.clientX, e.clientY);
+            } else if (e.rangeParent) {
+                range=document.createRange();
+                range.setStart(e.rangeParent,e.rangeOffset);
+            }
+            var sel = window.getSelection();
+            sel.removeAllRanges();
+            sel.addRange(range);
+            sel.anchorNode.parentNode.focus();
+
+            var html = (e.originalEvent || e).dataTransfer.getData('text/html');
+            var text = (e.originalEvent || e).dataTransfer.getData('text/plain');
+            var file = (e.originalEvent || e).dataTransfer.files;
+    
+            if (file.length) {
+                obj.addFile(file);
+            } else if (text) {
+                extractImageFromHtml(html);
+            }
+
+            el.classList.remove('jeditor-dragging');
+            e.preventDefault();
+        }
+    }
+
+    var editorBlur = function() {
+        obj.options.onblur(obj, el, obj.getData());
+    }
+
+    var editorFocus = function() {
+        obj.options.onfocus(obj, el, obj.getData());
+    }
+
+    editor.addEventListener('mouseup', editorMouseUp);
+    editor.addEventListener('mousedown', editorMouseDown);
+    editor.addEventListener('mousemove', editorMouseMove);
+    editor.addEventListener('keyup', editorKeyUp);
+    editor.addEventListener('keydown', editorKeyDown);
+    editor.addEventListener('dragstart', editorDragStart);
+    editor.addEventListener('dragenter', editorDragEnter);
+    editor.addEventListener('dragover', editorDragOver);
+    editor.addEventListener('drop', editorDrop);
+    editor.addEventListener('paste', editorPaste);
+
+    // Blur
+    if (typeof(obj.options.onblur) == 'function') {
+        editor.addEventListener('blur', editorBlur);
+    }
+
+    // Focus
+    if (typeof(obj.options.onfocus) == 'function') {
+        editor.addEventListener('focus', editorFocus);
+    }
+
+    // Onload
+    if (typeof(obj.options.onload) == 'function') {
+        obj.options.onload(el, editor);
+    }
+
+    // Set value to the editor
+    editor.innerHTML = value;
+
+    // Append editor to the containre
+    el.appendChild(editor);
+
+    // Snippet
+    if (obj.options.snippet) {
+        appendElement(obj.options.snippet);
+    }
+
+    // Default toolbar
+    if (obj.options.toolbar == null) {
+        obj.options.toolbar = jSuites.editor.getDefaultToolbar();
+    }
+
+    // Add toolbar
+    if (obj.options.toolbar) {
+        for (var i = 0; i < obj.options.toolbar.length; i++) {
+            if (obj.options.toolbar[i].icon) {
+                var item = document.createElement('div');
+                item.style.userSelect = 'none';
+                var itemIcon = document.createElement('i');
+                itemIcon.className = 'material-icons';
+                itemIcon.innerHTML = obj.options.toolbar[i].icon;
+                itemIcon.onclick = (function (a) {
+                    var b = a;
+                    return function () {
+                        obj.options.toolbar[b].onclick(el, obj, this)
+                    };
+                })(i);
+                item.appendChild(itemIcon);
+                toolbar.appendChild(item);
+            } else {
+                if (obj.options.toolbar[i].type == 'divisor') {
+                    var item = document.createElement('div');
+                    item.className = 'jeditor-toolbar-divisor';
+                    toolbar.appendChild(item);
+                } else if (obj.options.toolbar[i].type == 'button') {
+                    var item = document.createElement('div');
+                    item.classList.add('jeditor-toolbar-button');
+                    item.innerHTML = obj.options.toolbar[i].value;
+                    toolbar.appendChild(item);
+                }
+            }
+        }
+
+        el.appendChild(toolbar);
+    }
+
+    // Focus to the editor
+    if (obj.options.focus) {
+        jSuites.editor.setCursor(editor, obj.options.focus == 'initial' ? true : false);
+    }
+
+    el.editor = obj;
+
+    return obj;
+});
+
+jSuites.editor.setCursor = function(element, first) {
+    element.focus();
+    document.execCommand('selectAll');
+    var sel = window.getSelection();
+    var range = sel.getRangeAt(0);
+    if (first == true) {
+        var node = range.startContainer;
+        var size = 0;
+    } else {
+        var node = range.endContainer;
+        var size = node.length;
+    }
+    range.setStart(node, size);
+    range.setEnd(node, size);
+    sel.removeAllRanges();
+    sel.addRange(range);
+}
+
+jSuites.editor.getDomain = function(url) {
+    return url.replace('http://','').replace('https://','').replace('www.','').split(/[/?#]/)[0].split(/:/g)[0];
+}
+
+jSuites.editor.detectUrl = function(text) {
+    var expression = /(((https?:\/\/)|(www\.))[-A-Z0-9+&@#\/%?=~_|!:,.;]*[-A-Z0-9+&@#\/%=~_|]+)/ig;
+    var links = text.match(expression);
+
+    if (links) {
+        if (links[0].substr(0,3) == 'www') {
+            links[0] = 'http://' + links[0];
+        }
+    }
+
+    return links;
+}
+
+jSuites.editor.youtubeParser = function(url) {
+    var regExp = /^.*((youtu.be\/)|(v\/)|(\/u\/\w\/)|(embed\/)|(watch\?))\??v?=?([^#\&\?]*).*/;
+    var match = url.match(regExp);
+
+    return (match && match[7].length == 11) ? match[7] : false;
+}
+
+jSuites.editor.getDefaultToolbar = function() { 
+    return [
+        {
+            icon:'undo',
+            onclick: function() {
+                document.execCommand('undo');
+            }
+        },
+        {
+            icon:'redo',
+            onclick: function() {
+                document.execCommand('redo');
+            }
+        },
+        {
+            type:'divisor'
+        },
+        {
+            icon:'format_bold',
+            onclick: function(a,b,c) {
+                document.execCommand('bold');
+
+                if (document.queryCommandState("bold")) {
+                    c.classList.add('selected');
+                } else {
+                    c.classList.remove('selected');
+                }
+            }
+        },
+        {
+            icon:'format_italic',
+            onclick: function(a,b,c) {
+                document.execCommand('italic');
+
+                if (document.queryCommandState("italic")) {
+                    c.classList.add('selected');
+                } else {
+                    c.classList.remove('selected');
+                }
+            }
+        },
+        {
+            icon:'format_underline',
+            onclick: function(a,b,c) {
+                document.execCommand('underline');
+
+                if (document.queryCommandState("underline")) {
+                    c.classList.add('selected');
+                } else {
+                    c.classList.remove('selected');
+                }
+            }
+        },
+        {
+            type:'divisor'
+        },
+        {
+            icon:'format_list_bulleted',
+            onclick: function(a,b,c) {
+                document.execCommand('insertUnorderedList');
+
+                if (document.queryCommandState("insertUnorderedList")) {
+                    c.classList.add('selected');
+                } else {
+                    c.classList.remove('selected');
+                }
+            }
+        },
+        {
+            icon:'format_list_numbered',
+            onclick: function(a,b,c) {
+                document.execCommand('insertOrderedList');
+
+                if (document.queryCommandState("insertOrderedList")) {
+                    c.classList.add('selected');
+                } else {
+                    c.classList.remove('selected');
+                }
+            }
+        },
+        {
+            icon:'format_indent_increase',
+            onclick: function(a,b,c) {
+                document.execCommand('indent', true, null);
+
+                if (document.queryCommandState("indent")) {
+                    c.classList.add('selected');
+                } else {
+                    c.classList.remove('selected');
+                }
+            }
+        },
+        {
+            icon:'format_indent_decrease',
+            onclick: function(a,b,c) {
+                document.execCommand('outdent');
+
+                if (document.queryCommandState("outdent")) {
+                    c.classList.add('selected');
+                } else {
+                    c.classList.remove('selected');
+                }
+            }
+        }
+        /*{
+            type:'select',
+            items: ['Verdana','Arial','Courier New'],
+            onchange: function() {
+            }
+        },
+        {
+            type:'select',
+            items: ['10px','12px','14px','16px','18px','20px','22px'],
+            onchange: function() {
+            }
+        },
+        {
+            icon:'format_align_left',
+            onclick: function() {
+                document.execCommand('JustifyLeft');
+
+                if (document.queryCommandState("JustifyLeft")) {
+                    this.classList.add('selected');
+                } else {
+                    this.classList.remove('selected');
+                }
+            }
+        },
+        {
+            icon:'format_align_center',
+            onclick: function() {
+                document.execCommand('justifyCenter');
+
+                if (document.queryCommandState("justifyCenter")) {
+                    this.classList.add('selected');
+                } else {
+                    this.classList.remove('selected');
+                }
+            }
+        },
+        {
+            icon:'format_align_right',
+            onclick: function() {
+                document.execCommand('justifyRight');
+
+                if (document.queryCommandState("justifyRight")) {
+                    this.classList.add('selected');
+                } else {
+                    this.classList.remove('selected');
+                }
+            }
+        },
+        {
+            icon:'format_align_justify',
+            onclick: function() {
+                document.execCommand('justifyFull');
+
+                if (document.queryCommandState("justifyFull")) {
+                    this.classList.add('selected');
+                } else {
+                    this.classList.remove('selected');
+                }
+            }
+        },
+        {
+            icon:'format_list_bulleted',
+            onclick: function() {
+                document.execCommand('insertUnorderedList');
+
+                if (document.queryCommandState("insertUnorderedList")) {
+                    this.classList.add('selected');
+                } else {
+                    this.classList.remove('selected');
+                }
+            }
+        }*/
+    ];
+}
+
+
+jSuites.image = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // Default configuration
+    var defaults = {
+        minWidth: false,
+        onchange: null,
+        singleFile: true,
+        remoteParser: null,
+        text:{
+            extensionNotAllowed:'The extension is not allowed',
+            imageTooSmall:'The resolution is too low, try a image with a better resolution. width > 800px',
+        }
+    };
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Upload icon
+    el.classList.add('jupload');
+
+    // Add image
+    obj.addImage = function(file) {
+        if (! file.date) {
+            file.date = '';
+        }
+        var img = document.createElement('img');
+        img.setAttribute('data-date', file.lastmodified ? file.lastmodified : file.date);
+        img.setAttribute('data-name', file.name);
+        img.setAttribute('data-size', file.size);
+        img.setAttribute('data-small', file.small ? file.small : '');
+        img.setAttribute('data-cover', file.cover ? 1 : 0);
+        img.setAttribute('data-extension', file.extension);
+        img.setAttribute('src', file.file);
+        img.className = 'jfile';
+        img.style.width = '100%';
+
+        return img;
+    }
+
+    // Add image
+    obj.addImages = function(files) {
+        if (obj.options.singleFile == true) {
+            el.innerHTML = '';
+        }
+
+        for (var i = 0; i < files.length; i++) {
+            el.appendChild(obj.addImage(files[i]));
+        }
+    }
+
+    obj.addFromFile = function(file) {
+        var type = file.type.split('/');
+        if (type[0] == 'image') {
+            if (obj.options.singleFile == true) {
+                el.innerHTML = '';
+            }
+
+            var imageFile = new FileReader();
+            imageFile.addEventListener("load", function (v) {
+
+                var img = new Image();
+
+                img.onload = function onload() {
+                    var canvas = document.createElement('canvas');
+                    canvas.width = img.width;
+                    canvas.height = img.height;
+
+                    var ctx = canvas.getContext('2d');
+                    ctx.drawImage(img, 0, 0, canvas.width, canvas.height);
+
+                    var data = {
+                        file: canvas.toDataURL(),
+                        extension: file.name.substr(file.name.lastIndexOf('.') + 1),
+                        name: file.name,
+                        size: file.size,
+                        lastmodified: file.lastModified,
+                    }
+                    var newImage = obj.addImage(data);
+                    el.appendChild(newImage);
+
+                    // Onchange
+                    if (typeof(obj.options.onchange) == 'function') {
+                        obj.options.onchange(newImage);
+                    }
+                };
+
+                img.src = v.srcElement.result;
+            });
+
+            imageFile.readAsDataURL(file);
+        } else {
+            alert(text.extentionNotAllowed);
+        }
+    }
+
+    obj.addFromUrl = function(src) {
+        if (src.substr(0,4) != 'data' && ! obj.options.remoteParser) {
+            console.error('remoteParser not defined in your initialization');
+        } else {
+            // This is to process cross domain images
+            if (src.substr(0,4) == 'data') {
+                var extension = src.split(';')
+                extension = extension[0].split('/');
+                extension = extension[1];
+            } else {
+                var extension = src.substr(src.lastIndexOf('.') + 1);
+                // Work for cross browsers
+                src = obj.options.remoteParser + src;
+            }
+
+            var img = new Image();
+
+            img.onload = function onload() {
+                var canvas = document.createElement('canvas');
+                canvas.width = img.width;
+                canvas.height = img.height;
+
+                var ctx = canvas.getContext('2d');
+                ctx.drawImage(img, 0, 0, canvas.width, canvas.height);
+
+                canvas.toBlob(function(blob) {
+                    var data = {
+                        file: window.URL.createObjectURL(blob),
+                        extension: extension
+                    }
+                    var newImage = obj.addImage(data);
+                    el.appendChild(newImage);
+
+                    // Keep base64 ready to go
+                    var content = canvas.toDataURL();
+                    jSuites.files[data.file] = content.substr(content.indexOf(',') + 1);
+
+                    // Onchange
+                    if (typeof(obj.options.onchange) == 'function') {
+                        obj.options.onchange(newImage);
+                    }
+                });
+            };
+
+            img.src = src;
+        }
+    }
+
+    var attachmentInput = document.createElement('input');
+    attachmentInput.type = 'file';
+    attachmentInput.setAttribute('accept', 'image/*');
+    attachmentInput.onchange = function() {
+        for (var i = 0; i < this.files.length; i++) {
+            obj.addFromFile(this.files[i]);
+        }
+    }
+
+    el.addEventListener("dblclick", function(e) {
+        jSuites.click(attachmentInput);
+    });
+
+    el.addEventListener('dragenter', function(e) {
+        el.style.border = '1px dashed #000';
+    });
+
+    el.addEventListener('dragleave', function(e) {
+        el.style.border = '1px solid #eee';
+    });
+
+    el.addEventListener('dragstop', function(e) {
+        el.style.border = '1px solid #eee';
+    });
+
+    el.addEventListener('dragover', function(e) {
+        e.preventDefault();
+    });
+
+    el.addEventListener('drop', function(e) {
+        e.preventDefault();  
+        e.stopPropagation();
+
+
+        var html = (e.originalEvent || e).dataTransfer.getData('text/html');
+        var file = (e.originalEvent || e).dataTransfer.files;
+
+        if (file.length) {
+            for (var i = 0; i < e.dataTransfer.files.length; i++) {
+                obj.addFromFile(e.dataTransfer.files[i]);
+            }
+        } else if (html) {
+            if (obj.options.singleFile == true) {
+                el.innerHTML = '';
+            }
+
+            // Create temp element
+            var div = document.createElement('div');
+            div.innerHTML = html;
+
+            // Extract images
+            var img = div.querySelectorAll('img');
+
+            if (img.length) {
+                for (var i = 0; i < img.length; i++) {
+                    obj.addFromUrl(img[i].src);
+                }
+            }
+        }
+
+        el.style.border = '1px solid #eee';
+
+        return false;
+    });
+
+    el.image = obj;
+
+    return obj;
+});
+
+/**
+ * (c) jLoading
+ * https://github.com/paulhodel/jtools
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Page loading spin
+ */
+
+jSuites.loading = (function() {
+    var obj = {};
+
+    var loading = document.createElement('div');
+    loading.className = 'jloading';
+
+    obj.show = function() {
+        document.body.appendChild(loading);
+    };
+
+    obj.hide = function() {
+        document.body.removeChild(loading);
+    };
+
+    return obj;
+})();
+
+/**
+ * (c) jLogin
+ * https://github.com/paulhodel/jtools
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Login helper
+ */
+
+jSuites.login = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // Default configuration
+    var defaults = {
+        url: window.location.href,
+        prepareRequest: null,
+        accessToken: null,
+        deviceToken: null,
+        facebookUrl: null,
+        facebookAuthentication: null,
+        maxHeight: null,
+        onload: null,
+        onsuccess: null,
+        onerror: null,
+        message: null,
+        logo: null,
+        newProfile: false,
+        newProfileUrl: false,
+        newProfileLogin: false,
+        fullscreen: false,
+        newPasswordValidation: null,
+    };
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Message console container
+    if (! obj.options.message) {
+        var messageElement = document.querySelector('.message');
+        if (messageElement) {
+            obj.options.message = messageElement;
+        }
+    }
+
+    // Action
+    var action = null;
+
+    // Container
+    var container = document.createElement('form');
+    el.appendChild(container);
+
+    // Logo
+    var divLogo = document.createElement('div');
+    divLogo.className = 'jlogin-logo'
+    container.appendChild(divLogo);
+
+    if (obj.options.logo) {
+        var logo = document.createElement('img');
+        logo.src = obj.options.logo;
+        divLogo.appendChild(logo);
+    }
+
+    // Code
+    var labelCode = document.createElement('label');
+    labelCode.innerHTML = 'Please enter here the code received';
+    var inputCode = document.createElement('input');
+    inputCode.type = 'number';
+    inputCode.id = 'code';
+    inputCode.setAttribute('maxlength', 6);
+    var divCode = document.createElement('div');
+    divCode.appendChild(labelCode);
+    divCode.appendChild(inputCode);
+
+    // Hash
+    var inputHash = document.createElement('input');
+    inputHash.type = 'hidden';
+    inputHash.name = 'h';
+    var divHash = document.createElement('div');
+    divHash.appendChild(inputHash);
+
+    // Recovery
+    var inputRecovery = document.createElement('input');
+    inputRecovery.type = 'hidden';
+    inputRecovery.name = 'recovery';
+    inputRecovery.value = '1';
+    var divRecovery = document.createElement('div');
+    divRecovery.appendChild(inputRecovery);
+
+    // Login
+    var labelLogin = document.createElement('label');
+    labelLogin.innerHTML = 'Login';
+    var inputLogin = document.createElement('input');
+    inputLogin.type = 'text';
+    inputLogin.name = 'login';
+    inputLogin.setAttribute('autocomplete', 'off');
+    inputLogin.onkeyup = function() {
+        this.value = this.value.toLowerCase().replace(/[^a-zA-Z0-9_+]+/gi, '');
+    } 
+    var divLogin = document.createElement('div');
+    divLogin.appendChild(labelLogin);
+    divLogin.appendChild(inputLogin);
+
+    // Name
+    var labelName = document.createElement('label');
+    labelName.innerHTML = 'Name';
+    var inputName = document.createElement('input');
+    inputName.type = 'text';
+    inputName.name = 'name';
+    var divName = document.createElement('div');
+    divName.appendChild(labelName);
+    divName.appendChild(inputName);
+
+    // Email
+    var labelUsername = document.createElement('label');
+    labelUsername.innerHTML = 'E-mail';
+    var inputUsername = document.createElement('input');
+    inputUsername.type = 'text';
+    inputUsername.name = 'username';
+    inputUsername.setAttribute('autocomplete', 'new-username');
+    var divUsername = document.createElement('div');
+    divUsername.appendChild(labelUsername);
+    divUsername.appendChild(inputUsername);
+
+    // Password
+    var labelPassword = document.createElement('label');
+    labelPassword.innerHTML = 'New password';
+    var inputPassword = document.createElement('input');
+    inputPassword.type = 'password';
+    inputPassword.name = 'password';
+    inputPassword.setAttribute('autocomplete', 'new-password');
+    var divPassword = document.createElement('div');
+    divPassword.appendChild(labelPassword);
+    divPassword.appendChild(inputPassword);
+    divPassword.onkeydown = function(e) {
+        if (e.keyCode == 13) {
+            obj.execute();
+        }
+    }
+
+    // Repeat password
+    var labelRepeatPassword = document.createElement('label');
+    labelRepeatPassword.innerHTML = 'Repeat the new password';
+    var inputRepeatPassword = document.createElement('input');
+    inputRepeatPassword.type = 'password';
+    inputRepeatPassword.name = 'password';
+    var divRepeatPassword = document.createElement('div');
+    divRepeatPassword.appendChild(labelRepeatPassword);
+    divRepeatPassword.appendChild(inputRepeatPassword);
+
+    // Remember checkbox
+    var labelRemember = document.createElement('label');
+    labelRemember.innerHTML = 'Remember me on this device';
+    var inputRemember = document.createElement('input');
+    inputRemember.type = 'checkbox';
+    inputRemember.name = 'remember';
+    inputRemember.value = '1';
+    labelRemember.appendChild(inputRemember);
+    var divRememberButton = document.createElement('div');
+    divRememberButton.className = 'rememberButton';
+    divRememberButton.appendChild(labelRemember);
+
+    // Login button
+    var actionButton = document.createElement('input');
+    actionButton.type = 'button';
+    actionButton.value = 'Log In';
+    actionButton.onclick = function() {
+        obj.execute();
+    }
+    var divActionButton = document.createElement('div');
+    divActionButton.appendChild(actionButton);
+
+    // Cancel button
+    var cancelButton = document.createElement('div');
+    cancelButton.innerHTML = 'Cancel';
+    cancelButton.className = 'cancelButton';
+    cancelButton.onclick = function() {
+        obj.requestAccess();
+    }
+    var divCancelButton = document.createElement('div');
+    divCancelButton.appendChild(cancelButton);
+
+    // Captcha
+    var labelCaptcha = document.createElement('label');
+    labelCaptcha.innerHTML = 'Please type here the code below';
+    var inputCaptcha = document.createElement('input');
+    inputCaptcha.type = 'text';
+    inputCaptcha.name = 'captcha';
+    var imageCaptcha = document.createElement('img');
+    var divCaptcha = document.createElement('div');
+    divCaptcha.className = 'jlogin-captcha';
+    divCaptcha.appendChild(labelCaptcha);
+    divCaptcha.appendChild(inputCaptcha);
+    divCaptcha.appendChild(imageCaptcha);
+
+    // Facebook
+    var facebookButton = document.createElement('div');
+    facebookButton.innerHTML = 'Login with Facebook';
+    facebookButton.className = 'facebookButton';
+    var divFacebookButton = document.createElement('div');
+    divFacebookButton.appendChild(facebookButton);
+    divFacebookButton.onclick = function() {
+        obj.requestLoginViaFacebook();
+    }
+    // Forgot password
+    var inputRequest = document.createElement('span');
+    inputRequest.innerHTML = 'Request a new password';
+    var divRequestButton = document.createElement('div');
+    divRequestButton.className = 'requestButton';
+    divRequestButton.appendChild(inputRequest);
+    divRequestButton.onclick = function() {
+        obj.requestNewPassword();
+    }
+    // Create a new Profile
+    var inputNewProfile = document.createElement('span');
+    inputNewProfile.innerHTML = 'Create a new profile';
+    var divNewProfileButton = document.createElement('div');
+    divNewProfileButton.className = 'newProfileButton';
+    divNewProfileButton.appendChild(inputNewProfile);
+    divNewProfileButton.onclick = function() {
+        obj.newProfile();
+    }
+
+    el.className = 'jlogin';
+
+    if (obj.options.fullscreen == true) {
+        el.classList.add('jlogin-fullscreen');
+    }
+
+    /** 
+     * Show message
+     */
+    obj.showMessage = function(data) {
+        var message = (typeof(data) == 'object') ? data.message : data;
+
+        if (typeof(obj.options.showMessage) == 'function') {
+            obj.options.showMessage(data);
+        } else {
+            jSuites.alert(data);
+        }
+    }
+
+    /**
+     * New profile
+     */
+    obj.newProfile = function() {
+        container.innerHTML = '';
+        container.appendChild(divLogo);
+        if (obj.options.newProfileLogin) {
+            container.appendChild(divLogin);
+        }
+        container.appendChild(divName);
+        container.appendChild(divUsername);
+        container.appendChild(divActionButton);
+        if (obj.options.facebookAuthentication == true) {
+            container.appendChild(divFacebookButton);
+        }
+        container.appendChild(divCancelButton);
+
+        // Reset inputs
+        inputLogin.value = '';
+        inputUsername.value = '';
+        inputPassword.value = '';
+
+        // Button
+        actionButton.value = 'Create new profile';
+
+        // Action
+        action = 'newProfile';
+    }
+
+    /**
+     * Request the email with the recovery instructions
+     */
+    obj.requestNewPassword = function() {
+        if (Array.prototype.indexOf.call(container.children, divCaptcha) >= 0) {
+            var captcha = true;
+        }
+
+        container.innerHTML = '';
+        container.appendChild(divLogo);
+        container.appendChild(divRecovery);
+        container.appendChild(divUsername);
+        if (captcha) {
+            container.appendChild(divCaptcha);
+        }
+        container.appendChild(divActionButton);
+        container.appendChild(divCancelButton);
+        actionButton.value = 'Request a new password';
+        inputRecovery.value = 1;
+
+        // Action
+        action = 'requestNewPassword';
+    }
+
+    /**
+     * Confirm recovery code
+     */
+    obj.codeConfirmation = function() {
+        container.innerHTML = '';
+        container.appendChild(divLogo);
+        container.appendChild(divHash);
+        container.appendChild(divCode);
+        container.appendChild(divActionButton);
+        container.appendChild(divCancelButton);
+        actionButton.value = 'Confirm code';
+        inputRecovery.value = 2;
+
+        // Action
+        action = 'codeConfirmation';
+    }
+
+    /**
+     * Update my password
+     */
+    obj.changeMyPassword = function(hash) {
+        container.innerHTML = '';
+        container.appendChild(divLogo);
+        container.appendChild(divHash);
+        container.appendChild(divPassword);
+        container.appendChild(divRepeatPassword);
+        container.appendChild(divActionButton);
+        container.appendChild(divCancelButton);
+        actionButton.value = 'Change my password';
+        inputHash.value = hash;
+
+        // Action
+        action = 'changeMyPassword';
+    }
+
+    /**
+     * Request access default method
+     */
+    obj.requestAccess = function() {
+        container.innerHTML = '';
+        container.appendChild(divLogo);
+        container.appendChild(divUsername);
+        container.appendChild(divPassword);
+        container.appendChild(divActionButton);
+        if (obj.options.facebookAuthentication == true) {
+            container.appendChild(divFacebookButton);
+        }
+        container.appendChild(divRequestButton);
+        container.appendChild(divRememberButton);
+        container.appendChild(divRequestButton);
+        if (obj.options.newProfile == true) {
+            container.appendChild(divNewProfileButton);
+        }
+
+        // Button
+        actionButton.value = 'Login';
+
+        // Password
+        inputPassword.value = '';
+
+        // Email persistence
+        if (window.localStorage.getItem('username')) {
+            inputUsername.value = window.localStorage.getItem('username');
+            inputPassword.focus();
+        } else {
+            inputUsername.focus();
+        }
+
+        // Action
+        action = 'requestAccess';
+    }
+
+    /**
+     * Request login via facebook
+     */
+    obj.requestLoginViaFacebook = function() {
+        if (typeof(deviceNotificationToken) == 'undefined') {
+            FB.getLoginStatus(function(response) {
+                if (! response.status || response.status != 'connected') {
+                    FB.login(function(response) {
+                        if (response.authResponse) {
+                            obj.execute({ f:response.authResponse.accessToken });
+                        } else {
+                            obj.showMessage('Not authorized by facebook');
+                        }
+                    }, {scope: 'public_profile,email'});
+                } else {
+                    obj.execute({ f:response.authResponse.accessToken });
+                }
+            }, true);
+        } else {
+            jDestroy = function() {
+                fbLogin.removeEventListener('loadstart', jStart);
+                fbLogin.removeEventListener('loaderror', jError);
+                fbLogin.removeEventListener('exit', jExit);
+                fbLogin.close();
+                fbLogin = null;
+            }
+
+            jStart = function(event) {
+                var url = event.url;
+                if (url.indexOf("access_token") >= 0) {
+                    setTimeout(function(){
+                        var u = url.match(/=(.*?)&/);
+                        if (u[1].length > 32) {
+                            obj.execute({ f:u[1] });
+                        }
+                        jDestroy();
+                   },500);
+                }
+
+                if (url.indexOf("error=access_denied") >= 0) {
+                   setTimeout(jDestroy ,500);
+                   // Not authorized by facebook
+                   obj.showMessage('Not authorized by facebook');
+                }
+            }
+
+            jError = function(event) {
+                jDestroy();
+            }
+        
+            jExit = function(event) {
+                jDestroy();
+            }
+
+            fbLogin = window.open(obj.options.facebookUrl, "_blank", "location=no,closebuttoncaption=Exit,disallowoverscroll=yes,toolbar=no");
+            fbLogin.addEventListener('loadstart', jStart);
+            fbLogin.addEventListener('loaderror', jError);
+            fbLogin.addEventListener('exit', jExit);
+        }
+
+        // Action
+        action = 'requestLoginViaFacebook';
+    }
+
+    // Perform request
+    obj.execute = function(data) {
+        // New profile
+        if (action == 'newProfile') {
+            var pattern = new RegExp(/^([\w-\.]+@([\w-]+\.)+[\w-]{2,4})?$/);
+            if (! inputUsername.value || ! pattern.test(inputUsername.value)) {
+                var message = 'Invalid e-mail address'; 
+            }
+
+            var pattern = new RegExp(/^[a-zA-Z0-9\_\-\.\s+]+$/);
+            if (! inputLogin.value || ! pattern.test(inputLogin.value)) {
+                var message = 'Invalid username, please use only characters and numbers';
+            }
+
+            if (message) {
+                obj.showMessage(message);
+                return false;
+            }
+        } else if (action == 'changeMyPassword') {
+            if (inputPassword.value.length < 3) {
+                var message = 'Password is too short';
+            } else  if (inputPassword.value != inputRepeatPassword.value) {
+                var message = 'Password should match';
+            } else {
+                if (typeof(obj.options.newPasswordValidation) == 'function') {
+                    var val = obj.options.newPasswordValidation(obj, inputPassword.value, inputPassword.value);
+                    if (val != undefined) {
+                        message = val;
+                    }
+                }
+            }
+
+            if (message) {
+                obj.showMessage(message);
+                return false;
+            }
+        }
+
+        // Keep email
+        if (inputUsername.value != '') {
+            window.localStorage.setItem('username', inputUsername.value);
+        }
+
+        // Captcha
+        if (Array.prototype.indexOf.call(container.children, divCaptcha) >= 0) {
+            if (inputCaptcha.value == '') {
+                obj.showMessage('Please enter the captch code below');
+                return false;
+            }
+        }
+
+        // Url
+        var url = obj.options.url;
+
+        // Device token
+        if (obj.options.deviceToken) {
+            url += '?token=' + obj.options.deviceToken;
+        }
+
+        // Callback
+        var onsuccess = function(result) {
+            if (result) {
+                // Successfully response
+                if (result.success == 1) {
+                    // Recovery process
+                    if (action == 'requestNewPassword') {
+                        obj.codeConfirmation();
+                    } else if (action == 'codeConfirmation') {
+                        obj.requestAccess();
+                    } else if (action == 'newProfile') {
+                        obj.requestAccess();
+                        // New profile
+                        result.newProfile = true;
+                    }
+
+                    // Token
+                    if (result.token) {
+                        // Set token
+                        obj.options.accessToken = result.token;
+                        // Save token
+                        window.localStorage.setItem('Access-Token', result.token);
+                    }
+                }
+
+                // Show message
+                if (result.message) {
+                    // Show message
+                    obj.showMessage(result.message)
+                }
+
+                // Request captcha code
+                if (! result.data) {
+                    if (Array.prototype.indexOf.call(container.children, divCaptcha) >= 0) {
+                        divCaptcha.remove();
+                    }
+                } else {
+                    container.insertBefore(divCaptcha, divActionButton);
+                    imageCaptcha.setAttribute('src', 'data:image/png;base64,' + result.data);
+                }
+
+                // Give time to user see the message
+                if (result.hash) {
+                    // Change password
+                    obj.changeMyPassword(result.hash);
+                } else if (result.url) {
+                    // App initialization
+                    if (result.success == 1) {
+                        if (typeof(obj.options.onsuccess) == 'function') {
+                            obj.options.onsuccess(result);
+                        } else {
+                            if (result.message) {
+                                setTimeout(function() { window.location.href = result.url; }, 2000);
+                            } else {
+                                window.location.href = result.url;
+                            }
+                        }
+                    } else {
+                        if (typeof(obj.options.onerror) == 'function') {
+                            obj.options.onerror(result);
+                        }
+                    }
+                }
+            }
+        }
+
+        // Password
+        if (! data) {
+            var data = jSuites.getFormElements(el);
+            // Encode passworfd
+            if (data.password) {
+                data.password = jSuites.login.sha512(data.password);
+            }
+            // Recovery code
+            if (Array.prototype.indexOf.call(container.children, divCode) >= 0 && inputCode.value) {
+                data.h = jSuites.login.sha512(inputCode.value);
+            }
+        }
+
+        // Loading
+        el.classList.add('jlogin-loading');
+
+        // Url
+        var url = (action == 'newProfile' && obj.options.newProfileUrl) ? obj.options.newProfileUrl : obj.options.url;
+
+        // Remote call
+        jSuites.ajax({
+            url: url,
+            method: 'POST',
+            dataType: 'json',
+            data: data,
+            success: function(result) {
+                // Remove loading
+                el.classList.remove('jlogin-loading');
+                // Callback
+                onsuccess(result);
+            },
+            error: function(result) {
+                // Error
+                el.classList.remove('jlogin-loading');
+
+                if (typeof(obj.options.onerror) == 'function') {
+                    obj.options.onerror(result);
+                }
+            }
+        });
+    }
+
+    var queryString = window.location.href.split('?');
+    if (queryString[1] && queryString[1].length == 130 && queryString[1].substr(0,2) == 'h=') {
+        obj.changeMyPassword(queryString[1].substr(2));
+    } else {
+        obj.requestAccess();
+    }
+
+    return obj;
+});
+
+jSuites.login.sha512 = (function(str) {
+    function int64(msint_32, lsint_32) {
+        this.highOrder = msint_32;
+        this.lowOrder = lsint_32;
+    }
+
+    var H = [new int64(0x6a09e667, 0xf3bcc908), new int64(0xbb67ae85, 0x84caa73b),
+        new int64(0x3c6ef372, 0xfe94f82b), new int64(0xa54ff53a, 0x5f1d36f1),
+        new int64(0x510e527f, 0xade682d1), new int64(0x9b05688c, 0x2b3e6c1f),
+        new int64(0x1f83d9ab, 0xfb41bd6b), new int64(0x5be0cd19, 0x137e2179)];
+
+    var K = [new int64(0x428a2f98, 0xd728ae22), new int64(0x71374491, 0x23ef65cd),
+        new int64(0xb5c0fbcf, 0xec4d3b2f), new int64(0xe9b5dba5, 0x8189dbbc),
+        new int64(0x3956c25b, 0xf348b538), new int64(0x59f111f1, 0xb605d019),
+        new int64(0x923f82a4, 0xaf194f9b), new int64(0xab1c5ed5, 0xda6d8118),
+        new int64(0xd807aa98, 0xa3030242), new int64(0x12835b01, 0x45706fbe),
+        new int64(0x243185be, 0x4ee4b28c), new int64(0x550c7dc3, 0xd5ffb4e2),
+        new int64(0x72be5d74, 0xf27b896f), new int64(0x80deb1fe, 0x3b1696b1),
+        new int64(0x9bdc06a7, 0x25c71235), new int64(0xc19bf174, 0xcf692694),
+        new int64(0xe49b69c1, 0x9ef14ad2), new int64(0xefbe4786, 0x384f25e3),
+        new int64(0x0fc19dc6, 0x8b8cd5b5), new int64(0x240ca1cc, 0x77ac9c65),
+        new int64(0x2de92c6f, 0x592b0275), new int64(0x4a7484aa, 0x6ea6e483),
+        new int64(0x5cb0a9dc, 0xbd41fbd4), new int64(0x76f988da, 0x831153b5),
+        new int64(0x983e5152, 0xee66dfab), new int64(0xa831c66d, 0x2db43210),
+        new int64(0xb00327c8, 0x98fb213f), new int64(0xbf597fc7, 0xbeef0ee4),
+        new int64(0xc6e00bf3, 0x3da88fc2), new int64(0xd5a79147, 0x930aa725),
+        new int64(0x06ca6351, 0xe003826f), new int64(0x14292967, 0x0a0e6e70),
+        new int64(0x27b70a85, 0x46d22ffc), new int64(0x2e1b2138, 0x5c26c926),
+        new int64(0x4d2c6dfc, 0x5ac42aed), new int64(0x53380d13, 0x9d95b3df),
+        new int64(0x650a7354, 0x8baf63de), new int64(0x766a0abb, 0x3c77b2a8),
+        new int64(0x81c2c92e, 0x47edaee6), new int64(0x92722c85, 0x1482353b),
+        new int64(0xa2bfe8a1, 0x4cf10364), new int64(0xa81a664b, 0xbc423001),
+        new int64(0xc24b8b70, 0xd0f89791), new int64(0xc76c51a3, 0x0654be30),
+        new int64(0xd192e819, 0xd6ef5218), new int64(0xd6990624, 0x5565a910),
+        new int64(0xf40e3585, 0x5771202a), new int64(0x106aa070, 0x32bbd1b8),
+        new int64(0x19a4c116, 0xb8d2d0c8), new int64(0x1e376c08, 0x5141ab53),
+        new int64(0x2748774c, 0xdf8eeb99), new int64(0x34b0bcb5, 0xe19b48a8),
+        new int64(0x391c0cb3, 0xc5c95a63), new int64(0x4ed8aa4a, 0xe3418acb),
+        new int64(0x5b9cca4f, 0x7763e373), new int64(0x682e6ff3, 0xd6b2b8a3),
+        new int64(0x748f82ee, 0x5defb2fc), new int64(0x78a5636f, 0x43172f60),
+        new int64(0x84c87814, 0xa1f0ab72), new int64(0x8cc70208, 0x1a6439ec),
+        new int64(0x90befffa, 0x23631e28), new int64(0xa4506ceb, 0xde82bde9),
+        new int64(0xbef9a3f7, 0xb2c67915), new int64(0xc67178f2, 0xe372532b),
+        new int64(0xca273ece, 0xea26619c), new int64(0xd186b8c7, 0x21c0c207),
+        new int64(0xeada7dd6, 0xcde0eb1e), new int64(0xf57d4f7f, 0xee6ed178),
+        new int64(0x06f067aa, 0x72176fba), new int64(0x0a637dc5, 0xa2c898a6),
+        new int64(0x113f9804, 0xbef90dae), new int64(0x1b710b35, 0x131c471b),
+        new int64(0x28db77f5, 0x23047d84), new int64(0x32caab7b, 0x40c72493),
+        new int64(0x3c9ebe0a, 0x15c9bebc), new int64(0x431d67c4, 0x9c100d4c),
+        new int64(0x4cc5d4be, 0xcb3e42b6), new int64(0x597f299c, 0xfc657e2a),
+        new int64(0x5fcb6fab, 0x3ad6faec), new int64(0x6c44198c, 0x4a475817)];
+
+    var W = new Array(64);
+    var a, b, c, d, e, f, g, h, i, j;
+    var T1, T2;
+    var charsize = 8;
+
+    function utf8_encode(str) {
+        return unescape(encodeURIComponent(str));
+    }
+
+    function str2binb(str) {
+        var bin = [];
+        var mask = (1 << charsize) - 1;
+        var len = str.length * charsize;
+    
+        for (var i = 0; i < len; i += charsize) {
+            bin[i >> 5] |= (str.charCodeAt(i / charsize) & mask) << (32 - charsize - (i % 32));
+        }
+    
+        return bin;
+    }
+
+    function binb2hex(binarray) {
+        var hex_tab = "0123456789abcdef";
+        var str = "";
+        var length = binarray.length * 4;
+        var srcByte;
+
+        for (var i = 0; i < length; i += 1) {
+            srcByte = binarray[i >> 2] >> ((3 - (i % 4)) * 8);
+            str += hex_tab.charAt((srcByte >> 4) & 0xF) + hex_tab.charAt(srcByte & 0xF);
+        }
+
+        return str;
+    }
+
+    function safe_add_2(x, y) {
+        var lsw, msw, lowOrder, highOrder;
+
+        lsw = (x.lowOrder & 0xFFFF) + (y.lowOrder & 0xFFFF);
+        msw = (x.lowOrder >>> 16) + (y.lowOrder >>> 16) + (lsw >>> 16);
+        lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);
+
+        lsw = (x.highOrder & 0xFFFF) + (y.highOrder & 0xFFFF) + (msw >>> 16);
+        msw = (x.highOrder >>> 16) + (y.highOrder >>> 16) + (lsw >>> 16);
+        highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);
+
+        return new int64(highOrder, lowOrder);
+    }
+
+    function safe_add_4(a, b, c, d) {
+        var lsw, msw, lowOrder, highOrder;
+
+        lsw = (a.lowOrder & 0xFFFF) + (b.lowOrder & 0xFFFF) + (c.lowOrder & 0xFFFF) + (d.lowOrder & 0xFFFF);
+        msw = (a.lowOrder >>> 16) + (b.lowOrder >>> 16) + (c.lowOrder >>> 16) + (d.lowOrder >>> 16) + (lsw >>> 16);
+        lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);
+
+        lsw = (a.highOrder & 0xFFFF) + (b.highOrder & 0xFFFF) + (c.highOrder & 0xFFFF) + (d.highOrder & 0xFFFF) + (msw >>> 16);
+        msw = (a.highOrder >>> 16) + (b.highOrder >>> 16) + (c.highOrder >>> 16) + (d.highOrder >>> 16) + (lsw >>> 16);
+        highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);
+
+        return new int64(highOrder, lowOrder);
+    }
+
+    function safe_add_5(a, b, c, d, e) {
+        var lsw, msw, lowOrder, highOrder;
+
+        lsw = (a.lowOrder & 0xFFFF) + (b.lowOrder & 0xFFFF) + (c.lowOrder & 0xFFFF) + (d.lowOrder & 0xFFFF) + (e.lowOrder & 0xFFFF);
+        msw = (a.lowOrder >>> 16) + (b.lowOrder >>> 16) + (c.lowOrder >>> 16) + (d.lowOrder >>> 16) + (e.lowOrder >>> 16) + (lsw >>> 16);
+        lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);
+
+        lsw = (a.highOrder & 0xFFFF) + (b.highOrder & 0xFFFF) + (c.highOrder & 0xFFFF) + (d.highOrder & 0xFFFF) + (e.highOrder & 0xFFFF) + (msw >>> 16);
+        msw = (a.highOrder >>> 16) + (b.highOrder >>> 16) + (c.highOrder >>> 16) + (d.highOrder >>> 16) + (e.highOrder >>> 16) + (lsw >>> 16);
+        highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);
+
+        return new int64(highOrder, lowOrder);
+    }
+
+    function maj(x, y, z) {
+        return new int64(
+            (x.highOrder & y.highOrder) ^ (x.highOrder & z.highOrder) ^ (y.highOrder & z.highOrder),
+            (x.lowOrder & y.lowOrder) ^ (x.lowOrder & z.lowOrder) ^ (y.lowOrder & z.lowOrder)
+        );
+    }
+
+    function ch(x, y, z) {
+        return new int64(
+            (x.highOrder & y.highOrder) ^ (~x.highOrder & z.highOrder),
+            (x.lowOrder & y.lowOrder) ^ (~x.lowOrder & z.lowOrder)
+        );
+    }
+
+    function rotr(x, n) {
+        if (n <= 32) {
+            return new int64(
+             (x.highOrder >>> n) | (x.lowOrder << (32 - n)),
+             (x.lowOrder >>> n) | (x.highOrder << (32 - n))
+            );
+        } else {
+            return new int64(
+             (x.lowOrder >>> n) | (x.highOrder << (32 - n)),
+             (x.highOrder >>> n) | (x.lowOrder << (32 - n))
+            );
+        }
+    }
+
+    function sigma0(x) {
+        var rotr28 = rotr(x, 28);
+        var rotr34 = rotr(x, 34);
+        var rotr39 = rotr(x, 39);
+
+        return new int64(
+            rotr28.highOrder ^ rotr34.highOrder ^ rotr39.highOrder,
+            rotr28.lowOrder ^ rotr34.lowOrder ^ rotr39.lowOrder
+        );
+    }
+
+    function sigma1(x) {
+        var rotr14 = rotr(x, 14);
+        var rotr18 = rotr(x, 18);
+        var rotr41 = rotr(x, 41);
+
+        return new int64(
+            rotr14.highOrder ^ rotr18.highOrder ^ rotr41.highOrder,
+            rotr14.lowOrder ^ rotr18.lowOrder ^ rotr41.lowOrder
+        );
+    }
+
+    function gamma0(x) {
+        var rotr1 = rotr(x, 1), rotr8 = rotr(x, 8), shr7 = shr(x, 7);
+
+        return new int64(
+            rotr1.highOrder ^ rotr8.highOrder ^ shr7.highOrder,
+            rotr1.lowOrder ^ rotr8.lowOrder ^ shr7.lowOrder
+        );
+    }
+
+    function gamma1(x) {
+        var rotr19 = rotr(x, 19);
+        var rotr61 = rotr(x, 61);
+        var shr6 = shr(x, 6);
+
+        return new int64(
+            rotr19.highOrder ^ rotr61.highOrder ^ shr6.highOrder,
+            rotr19.lowOrder ^ rotr61.lowOrder ^ shr6.lowOrder
+        );
+    }
+
+    function shr(x, n) {
+        if (n <= 32) {
+            return new int64(
+                x.highOrder >>> n,
+                x.lowOrder >>> n | (x.highOrder << (32 - n))
+            );
+        } else {
+            return new int64(
+                0,
+                x.highOrder << (32 - n)
+            );
+        }
+    }
+
+    var str = utf8_encode(str);
+    var strlen = str.length*charsize;
+    str = str2binb(str);
+
+    str[strlen >> 5] |= 0x80 << (24 - strlen % 32);
+    str[(((strlen + 128) >> 10) << 5) + 31] = strlen;
+
+    for (var i = 0; i < str.length; i += 32) {
+        a = H[0];
+        b = H[1];
+        c = H[2];
+        d = H[3];
+        e = H[4];
+        f = H[5];
+        g = H[6];
+        h = H[7];
+
+        for (var j = 0; j < 80; j++) {
+            if (j < 16) {
+                W[j] = new int64(str[j*2 + i], str[j*2 + i + 1]);
+            } else {
+                W[j] = safe_add_4(gamma1(W[j - 2]), W[j - 7], gamma0(W[j - 15]), W[j - 16]);
+            }
+
+            T1 = safe_add_5(h, sigma1(e), ch(e, f, g), K[j], W[j]);
+            T2 = safe_add_2(sigma0(a), maj(a, b, c));
+            h = g;
+            g = f;
+            f = e;
+            e = safe_add_2(d, T1);
+            d = c;
+            c = b;
+            b = a;
+            a = safe_add_2(T1, T2);
+        }
+
+        H[0] = safe_add_2(a, H[0]);
+        H[1] = safe_add_2(b, H[1]);
+        H[2] = safe_add_2(c, H[2]);
+        H[3] = safe_add_2(d, H[3]);
+        H[4] = safe_add_2(e, H[4]);
+        H[5] = safe_add_2(f, H[5]);
+        H[6] = safe_add_2(g, H[6]);
+        H[7] = safe_add_2(h, H[7]);
+    }
+
+    var binarray = [];
+    for (var i = 0; i < H.length; i++) {
+        binarray.push(H[i].highOrder);
+        binarray.push(H[i].lowOrder);
+    }
+
+    return binb2hex(binarray);
+});
+
+/**
+ * (c) jTools Input Mask
+ * https://github.com/paulhodel/jtools
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Input mask
+ */
+
+jSuites.mask = (function() {
+    var obj = {};
+    var index = 0;
+    var values = []
+    var pieces = [];
+
+    obj.run = function(value, mask, decimal) {
+        if (value && mask) {
+            if (! decimal) {
+                decimal = '.';
+            }
+            if (value == Number(value)) {
+                var number = (''+value).split('.');
+                var value = number[0];
+                var valueDecimal = number[1];
+            } else {
+                value = '' + value;
+            }
+            index = 0;
+            values = [];
+            // Create mask token
+            obj.prepare(mask);
+            // Current value
+            var currentValue = value;
+            if (currentValue) {
+                // Checking current value
+                for (var i = 0; i < currentValue.length; i++) {
+                    if (currentValue[i] != null) {
+                        obj.process(currentValue[i]);
+                    }
+                }
+            }
+            if (valueDecimal) {
+                obj.process(decimal);
+                var currentValue = valueDecimal;
+                if (currentValue) {
+                    // Checking current value
+                    for (var i = 0; i < currentValue.length; i++) {
+                        if (currentValue[i] != null) {
+                            obj.process(currentValue[i]);
+                        }
+                    }
+                }
+            }
+            // Formatted value
+            return values.join('');
+        } else {
+            return '';
+        }
+    }
+
+    obj.apply = function(e) {
+        if (e.target && ! e.target.getAttribute('readonly')) {
+            var mask = e.target.getAttribute('data-mask');
+            if (mask && e.keyCode > 46) {
+                index = 0;
+                values = [];
+                // Create mask token
+                obj.prepare(mask);
+                // Current value
+                if (e.target.selectionStart < e.target.selectionEnd) {
+                    var currentValue = e.target.value.substring(0, e.target.selectionStart); 
+                } else {
+                    var currentValue = e.target.value;
+                }
+                if (currentValue) {
+                    // Checking current value
+                    for (var i = 0; i < currentValue.length; i++) {
+                        if (currentValue[i] != null) {
+                            obj.process(currentValue[i]);
+                        }
+                    }
+                }
+                // New input
+                obj.process(obj.fromKeyCode(e));
+                // Update value to the element
+                e.target.value = values.join('');
+                if (pieces.length == values.length && pieces[pieces.length-1].length == values[values.length-1].length) {
+                    e.target.setAttribute('data-completed', 'true');
+                } else {
+                    e.target.setAttribute('data-completed', 'false');
+                }
+                // Prevent default
+                e.preventDefault();
+            }
+        }
+    }
+
+    /**
+     * Process inputs and save to values
+     */
+    obj.process = function(input) {
+        do {
+            if (pieces[index] == 'mm') {
+                if (values[index] == null || values[index] == '') {
+                    if (parseInt(input) > 1 && parseInt(input) < 10) {
+                        values[index] = '0' + input;
+                        index++;
+                        return true;
+                    } else if (parseInt(input) < 10) {
+                        values[index] = input;
+                        return true;
+                    } else {
+                        return false;
+                    }
+                } else {
+                    if (values[index] == 1 && values[index] < 2 && parseInt(input) < 3) {
+                        values[index] += input;
+                        index++;
+                        return true;
+                    } else if (values[index] == 0 && values[index] < 10) {
+                        values[index] += input;
+                        index++;
+                        return true;
+                    } else {
+                        return false
+                    }
+                }
+            } else if (pieces[index] == 'dd') {
+                if (values[index] == null || values[index] == '') {
+                    if (parseInt(input) > 3 && parseInt(input) < 10) {
+                        values[index] = '0' + input;
+                        index++;
+                        return true;
+                    } else if (parseInt(input) < 10) {
+                        values[index] = input;
+                        return true;
+                    } else {
+                        return false;
+                    }
+                } else {
+                    if (values[index] == 3 && parseInt(input) < 2) {
+                        values[index] += input;
+                        index++;
+                        return true;
+                    } else if (values[index] < 3 && parseInt(input) < 10) {
+                        values[index] += input;
+                        index++;
+                        return true;
+                    } else {
+                        return false
+                    }
+                }
+            } else if (pieces[index] == 'hh24') {
+                if (values[index] == null || values[index] == '') {
+                    if (parseInt(input) > 2 && parseInt(input) < 10) {
+                        values[index] = '0' + input;
+                        index++;
+                        return true;
+                    } else if (parseInt(input) < 10) {
+                        values[index] = input;
+                        return true;
+                    } else {
+                        return false;
+                    }
+                } else {
+                    if (values[index] == 2 && parseInt(input) < 4) {
+                        values[index] += input;
+                        index++;
+                        return true;
+                    } else if (values[index] < 2 && parseInt(input) < 10) {
+                        values[index] += input;
+                        index++;
+                        return true;
+                    } else {
+                        return false
+                    }
+                }
+            } else if (pieces[index] == 'hh') {
+                if (values[index] == null || values[index] == '') {
+                    if (parseInt(input) > 1 && parseInt(input) < 10) {
+                        values[index] = '0' + input;
+                        index++;
+                        return true;
+                    } else if (parseInt(input) < 10) {
+                        values[index] = input;
+                        return true;
+                    } else {
+                        return false;
+                    }
+                } else {
+                    if (values[index] == 1 && parseInt(input) < 3) {
+                        values[index] += input;
+                        index++;
+                        return true;
+                    } else if (values[index] < 1 && parseInt(input) < 10) {
+                        values[index] += input;
+                        index++;
+                        return true;
+                    } else {
+                        return false
+                    }
+                }
+            } else if (pieces[index] == 'mi' || pieces[index] == 'ss') {
+                if (values[index] == null || values[index] == '') {
+                    if (parseInt(input) > 5 && parseInt(input) < 10) {
+                        values[index] = '0' + input;
+                        index++;
+                        return true;
+                    } else if (parseInt(input) < 10) {
+                        values[index] = input;
+                        return true;
+                    } else {
+                        return false;
+                    }
+                } else {
+                    if (parseInt(input) < 10) {
+                        values[index] += input;
+                        index++;
+                        return true;
+                     } else {
+                        return false
+                    }
+                }
+            } else if (pieces[index] == 'yy' || pieces[index] == 'yyyy') {
+                if (parseInt(input) < 10) {
+                    if (values[index] == null || values[index] == '') {
+                        values[index] = input;
+                    } else {
+                        values[index] += input;
+                    }
+                    
+                    if (values[index].length == pieces[index].length) {
+                        index++;
+                    }
+                    return true;
+                } else {
+                    return false;
+                }
+            } else if (pieces[index] == '#' || pieces[index] == '#.##' || pieces[index] == '#,##' || pieces[index] == '# ##') {
+                if (input.match(/[0-9]/g)) {
+                    if (pieces[index] == '#.##') {
+                        var separator = '.';
+                    } else if (pieces[index] == '#,##') {
+                        var separator = ',';
+                    } else if (pieces[index] == '# ##') {
+                        var separator = ' ';
+                    } else {
+                        var separator = '';
+                    }
+                    if (values[index] == null || values[index] == '') {
+                        values[index] = input;
+                    } else {
+                        values[index] += input;
+                        if (separator) {
+                            values[index] = values[index].match(/[0-9]/g).join('');
+                            var t = [];
+                            var s = 0;
+                            for (var j = values[index].length - 1; j >= 0 ; j--) {
+                                t.push(values[index][j]);
+                                s++;
+                                if (! (s % 3)) {
+                                    t.push(separator);
+                                }
+                            }
+                            t = t.reverse();
+                            values[index] = t.join('');
+                            if (values[index].substr(0,1) == separator) {
+                                values[index] = values[index].substr(1);
+                            } 
+                        }
+                    }
+                    return true;
+                } else {
+                    if (pieces[index] == '#.##' && input == '.') {
+                        // Do nothing
+                    } else if (pieces[index] == '#,##' && input == ',') {
+                        // Do nothing
+                    } else if (pieces[index] == '# ##' && input == ' ') {
+                        // Do nothing
+                    } else {
+                        if (values[index]) {
+                            index++;
+                            if (pieces[index]) {
+                                if (pieces[index] == input) {
+                                    values[index] = input;
+                                    return true;
+                                } else {
+                                    if (pieces[index] == '0' && pieces[index+1] == input) {
+                                        index++;
+                                        values[index] = input;
+                                        return true;
+                                    }
+                                }
+                            }
+                        }
+                    }
+
+                    return false;
+                }
+            } else if (pieces[index] == '0') {
+                if (input.match(/[0-9]/g)) {
+                    values[index] = input;
+                    index++;
+                    return true;
+                } else {
+                    return false;
+                }
+            } else if (pieces[index] == 'a') {
+                if (input.match(/[a-zA-Z]/g)) {
+                    values[index] = input;
+                    index++;
+                    return true;
+                } else {
+                    return false;
+                }
+            } else {
+                if (pieces[index] != null) {
+                    if (pieces[index] == '\\a') {
+                        var v = 'a';
+                    } else if (pieces[index] == '\\0') {
+                        var v = '0';
+                    } else if (pieces[index] == '[-]') {
+                        if (input == '-' || input == '+') {
+                            var v = input;
+                        } else {
+                            var v = ' ';
+                        }
+                    } else {
+                        var v = pieces[index];
+                    }
+                    values[index] = v;
+                    if (input == v) {
+                        index++;
+                        return true;
+                    }
+                }
+            }
+
+            index++;
+        } while (pieces[index]);
+    }
+
+    /**
+     * Create tokens for the mask
+     */
+    obj.prepare = function(mask) {
+        pieces = [];
+        for (var i = 0; i < mask.length; i++) {
+            if (mask[i].match(/[0-9]|[a-z]|\\/g)) {
+                if (mask[i] == 'y' && mask[i+1] == 'y' && mask[i+2] == 'y' && mask[i+3] == 'y') {
+                    pieces.push('yyyy');
+                    i += 3;
+                } else if (mask[i] == 'y' && mask[i+1] == 'y') {
+                    pieces.push('yy');
+                    i++;
+                } else if (mask[i] == 'm' && mask[i+1] == 'm' && mask[i+2] == 'm' && mask[i+3] == 'm') {
+                    pieces.push('mmmm');
+                    i += 3;
+                } else if (mask[i] == 'm' && mask[i+1] == 'm' && mask[i+2] == 'm') {
+                    pieces.push('mmm');
+                    i += 2;
+                } else if (mask[i] == 'm' && mask[i+1] == 'm') {
+                    pieces.push('mm');
+                    i++;
+                } else if (mask[i] == 'd' && mask[i+1] == 'd') {
+                    pieces.push('dd');
+                    i++;
+                } else if (mask[i] == 'h' && mask[i+1] == 'h' && mask[i+2] == '2' && mask[i+3] == '4') {
+                    pieces.push('hh24');
+                    i += 3;
+                } else if (mask[i] == 'h' && mask[i+1] == 'h') {
+                    pieces.push('hh');
+                    i++;
+                } else if (mask[i] == 'm' && mask[i+1] == 'i') {
+                    pieces.push('mi');
+                    i++;
+                } else if (mask[i] == 's' && mask[i+1] == 's') {
+                    pieces.push('ss');
+                    i++;
+                } else if (mask[i] == 'a' && mask[i+1] == 'm') {
+                    pieces.push('am');
+                    i++;
+                } else if (mask[i] == 'p' && mask[i+1] == 'm') {
+                    pieces.push('pm');
+                    i++;
+                } else if (mask[i] == '\\' && mask[i+1] == '0') {
+                    pieces.push('\\0');
+                    i++;
+                } else if (mask[i] == '\\' && mask[i+1] == 'a') {
+                    pieces.push('\\a');
+                    i++;
+                } else {
+                    pieces.push(mask[i]);
+                }
+            } else {
+                if (mask[i] == '#' && mask[i+1] == '.' && mask[i+2] == '#' && mask[i+3] == '#') {
+                    pieces.push('#.##');
+                    i += 3;
+                } else if (mask[i] == '#' && mask[i+1] == ',' && mask[i+2] == '#' && mask[i+3] == '#') {
+                    pieces.push('#,##');
+                    i += 3;
+                } else if (mask[i] == '#' && mask[i+1] == ' ' && mask[i+2] == '#' && mask[i+3] == '#') {
+                    pieces.push('# ##');
+                    i += 3;
+                } else if (mask[i] == '[' && mask[i+1] == '-' && mask[i+2] == ']') {
+                    pieces.push('[-]');
+                    i += 2;
+                } else {
+                    pieces.push(mask[i]);
+                }
+            }
+        }
+    }
+
+    /** 
+     * Thanks for the collaboration
+     */
+    obj.fromKeyCode = function(e) {
+        var _to_ascii = {
+            '188': '44',
+            '109': '45',
+            '190': '46',
+            '191': '47',
+            '192': '96',
+            '220': '92',
+            '222': '39',
+            '221': '93',
+            '219': '91',
+            '173': '45',
+            '187': '61', //IE Key codes
+            '186': '59', //IE Key codes
+            '189': '45'  //IE Key codes
+        }
+
+        var shiftUps = {
+            "96": "~",
+            "49": "!",
+            "50": "@",
+            "51": "#",
+            "52": "$",
+            "53": "%",
+            "54": "^",
+            "55": "&",
+            "56": "*",
+            "57": "(",
+            "48": ")",
+            "45": "_",
+            "61": "+",
+            "91": "{",
+            "93": "}",
+            "92": "|",
+            "59": ":",
+            "39": "\"",
+            "44": "<",
+            "46": ">",
+            "47": "?"
+        };
+
+        var c = e.which;
+
+        if (_to_ascii.hasOwnProperty(c)) {
+            c = _to_ascii[c];
+        }
+
+        if (!e.shiftKey && (c >= 65 && c <= 90)) {
+            c = String.fromCharCode(c + 32);
+        } else if (e.shiftKey && shiftUps.hasOwnProperty(c)) {
+            c = shiftUps[c];
+        } else if (96 <= c && c <= 105) {
+            c = String.fromCharCode(c - 48);
+        } else {
+            c = String.fromCharCode(c);
+        }
+
+        return c;
+    }
+
+    return obj;
+})();
+
+jSuites.mobile = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    if (jSuites.el) {
+        jSuites.el.addEventListener('mousedown', function(e) {
+            if (e.target.classList.contains('option-title')) {
+                if (e.target.classList.contains('selected')) {
+                    e.target.classList.remove('selected');
+                } else {
+                    e.target.classList.add('selected');
+                }
+            }
+        });
+    }
+
+    return obj;
+})();
+
+jSuites.pages = (function() {
+    var container = null;
+    var current = null;
+
+    // Create a page
+    var createPage = function(options, callback) {
+        // Create page
+        var page = document.createElement('div');
+        page.classList.add('page');
+
+        // Always hidden
+        page.style.display = 'none';
+
+        // Keep options
+        page.options = options ? options : {};
+
+        if (! current) {
+            container.appendChild(page);
+        } else {
+            container.insertBefore(page, current.nextSibling);
+        }
+
+        jSuites.ajax({
+            url: page.options.url,
+            method: 'GET',
+            success: function(result) {
+                // Push to refresh controls
+                jSuites.refresh(page, page.options.onpush);
+
+                // Open page
+                page.innerHTML = result;
+                // Get javascript
+                var script = page.getElementsByTagName('script');
+                // Run possible inline scripts
+                for (var i = 0; i < script.length; i++) {
+                    // Get type
+                    var type = script[i].getAttribute('type');
+                    if (! type || type == 'text/javascript') {
+                        eval(script[i].innerHTML);
+                    }
+                }
+                // Set title
+                page.setTitle = function(text) {
+                    this.children[0].children[0].children[1].innerHTML = text;
+                }
+                // Show page
+                if (! page.options.closed) {
+                    showPage(page);
+                }
+                // Onload callback
+                if (typeof(page.options.onload) == 'function') {
+                    page.options.onload(page);
+                }
+                // Force callback
+                if (typeof(callback) == 'function') {
+                    callback(page);
+                }
+            }
+        });
+
+        return page;
+    }
+
+    var showPage = function(page, ignoreHistory, callback) {
+        if (current) {
+            if (current == page) {
+                current = page;
+            } else {
+                // Keep scroll in the top
+                window.scrollTo({ top: 0 });
+
+                // Show page
+                page.style.display = '';
+
+                var a = Array.prototype.indexOf.call(container.children, current);
+                var b = Array.prototype.indexOf.call(container.children, page);
+
+                // Leave
+                if (typeof(current.options.onleave) == 'function') {
+                    current.options.onleave(current, page, ignoreHistory);
+                }
+
+                jSuites.slideLeft(container, (a < b ? 0 : 1), function() {
+                    current.style.display = 'none';
+                    current = page;
+                });
+
+                // Enter
+                if (typeof(page.options.onenter) == 'function') {
+                    page.options.onenter(page, current, ignoreHistory);
+                }
+            }
+        } else {
+            // Show
+            page.style.display = '';
+
+            // Keep current
+            current = page;
+
+            // Enter
+            if (typeof(page.options.onenter) == 'function') {
+                page.options.onenter(page);
+            }
+        }
+
+        // Add history
+        if (! ignoreHistory) {
+            // Add history
+            window.history.pushState({ route: page.options.route }, page.options.title, page.options.route);
+        }
+
+        // Callback
+        if (typeof(callback) == 'function') {
+            callback(page);
+        }
+    }
+
+    // Init method
+    var obj = function(route, mixed) {
+
+        // Create page container
+        if (! container) {
+            container = document.querySelector('.pages');
+            if (! container) {
+                container = document.createElement('div');
+                container.className = 'pages';
+            }
+
+            // Append container to the application
+            if (jSuites.el) {
+                jSuites.el.appendChild(container);
+            } else {
+                document.body.appendChild(container);
+            }
+        }
+
+        if (! obj.pages[route]) {
+            if (! route) {
+                alert('Error, no route provided');
+            } else {
+                if (typeof(mixed) == 'function') {
+                    var options = {};
+                    var callback = mixed;
+                } else {
+                    // Page options
+                    var options = mixed ? mixed : {};
+                }
+
+                // Closed
+                options.closed = mixed && mixed.closed ? 1 : 0;
+                // Keep Route
+                options.route = route;
+
+                // New page url
+                if (! options.url) {
+                    var routePath = route.split('#');
+                    options.url = jSuites.pages.path + routePath[0] + '.html';
+                }
+                // Title
+                if (! options.title) {
+                    options.title = 'Untitled';
+                }
+
+                // Create new page
+                obj.pages[route] = createPage(options, callback ? callback : null);
+            }
+        } else {
+            // Update config
+            if (mixed) {
+                // History
+                var ignoreHistory = 0;
+
+                if (typeof(mixed) == 'function') {
+                    var callback = mixed;
+                } else {
+                    if (typeof(mixed.onenter) == 'function') {
+                        obj.pages[route].options.onenter = mixed.onenter;
+                    }
+                    if (typeof(mixed.onleave) == 'function') {
+                        obj.pages[route].options.onleave = mixed.onleave;
+                    }
+
+                    // Ignore history
+                    ignoreHistory = mixed.ignoreHistory ? 1 : 0; 
+                }
+            }
+
+            showPage(obj.pages[route], ignoreHistory, callback ? callback : null);
+        }
+    }
+
+    obj.pages = {};
+
+    // Get page
+    obj.get = function(route) {
+        if (obj.pages[route]) {
+            return obj.pages[route]; 
+        }
+    }
+
+    obj.getContainer = function() {
+        return container;
+    }
+
+    obj.destroy = function() {
+        // Current is null
+        current = null;
+        // Destroy containers
+        obj.pages = {};
+        // Reset container
+        if (container) {
+            container.innerHTML = '';
+        }
+    }
+
+    return obj;
+})();
+
+// Path
+jSuites.pages.path = 'pages';
+
+// Panel
+jSuites.panel = (function() {
+    // No initial panel declared
+    var panel = null;
+
+    var obj = function(route) {
+        if (! panel) {
+            obj.create(jSuites.pages.path + route + '.html');
+        }
+
+        // Show panel
+        panel.style.display = '';
+
+        // Add animation
+        if (panel.classList.contains('panel-left')) {
+            jSuites.slideLeft(panel, 1);
+        } else {
+            jSuites.slideRight(panel, 1);
+        }
+    }
+
+    obj.create = function(route) {
+        if (! panel) {
+            // Create element
+            panel = document.createElement('div');
+            panel.classList.add('panel');
+            panel.classList.add('panel-left');
+            panel.style.display = 'none';
+
+            // Bind to the app
+            if (jSuites.el) {
+                jSuites.el.appendChild(panel);
+            } else {
+                document.body.appendChild(panel);
+            }
+        }
+
+        // Remote content
+        if (route) {
+	        var url = jSuites.pages.path + route + '.html';
+
+            jSuites.ajax({
+                url: url,
+                method: 'GET',
+                success: function(result) {
+                    // Set content
+                    panel.innerHTML = result;
+                    // Get javascript
+                    var script = panel.getElementsByTagName('script');
+                    // Run possible inline scripts
+                    for (var i = 0; i < script.length; i++) {
+                        // Get type
+                        var type = script[i].getAttribute('type');
+                        if (! type || type == 'text/javascript') {
+                            eval(script[i].innerHTML);
+                        }
+                    }
+                }
+            });
+        }
+    }
+
+    obj.close = function() {
+        if (panel) {
+            // Animation
+            if (panel.classList.contains('panel-left')) {
+                jSuites.slideLeft(panel, 0, function() {
+                    panel.style.display = 'none';
+                });
+            } else {
+                jSuites.slideRight(panel, 0, function() {
+                    panel.style.display = 'none';
+                });
+            }
+        }
+    }
+
+    obj.get = function() {
+        return panel;
+    }
+
+    obj.destroy = function() {
+        panel.remove();
+        panel = null;
+    }
+
+    return obj;
+})();
+
+jSuites.toolbar = (function(el, options) {
+    var obj = {};
+    obj.options = options;
+
+    obj.selectItem = function(element) {
+        var elements = toolbarContent.children;
+        for (var i = 0; i < elements.length; i++) {
+            elements[i].classList.remove('selected');
+        }
+        element.classList.add('selected');
+    }
+
+    obj.hide = function() {
+        jSuites.slideBottom(toolbar, 0, function() {
+            toolbar.style.display = 'none';
+        });
+    }
+
+    obj.show = function() {
+        toolbar.style.display = '';
+        jSuites.slideBottom(toolbar, 1);
+    }
+
+    obj.get = function() {
+        return toolbar;
+    }
+
+    obj.setBadge = function(index, value) {
+        toolbarContent.children[index].children[1].firstChild.innerHTML = value;
+    }
+
+    obj.destroy = function() {
+        toolbar.remove();
+        toolbar = null;
+    }
+
+    var toolbar = document.createElement('div');
+    toolbar.classList.add('jtoolbar');
+    toolbar.onclick = function(e) {
+        var element = jSuites.getElement(e.target, 'jtoolbar-item');
+        if (element) {
+            obj.selectItem(element);
+        }
+    }
+
+    var toolbarContent = document.createElement('div');
+    toolbar.appendChild(toolbarContent);
+
+    for (var i = 0; i < options.items.length; i++) {
+        var toolbarItem = document.createElement('div');
+        toolbarItem.classList.add('jtoolbar-item');
+        if (options.items[i].route) {
+            toolbarItem.setAttribute('data-href', options.items[i].route);
+            jSuites.pages(options.items[i].route, {
+                closed: true,
+                onenter: function() {
+                    obj.selectItem(toolbarItem);
+                }
+            });
+        }
+
+        if (options.items[i].icon) {
+            var toolbarIcon = document.createElement('i');
+            toolbarIcon.classList.add('material-icons');
+            toolbarIcon.innerHTML = options.items[i].icon;
+            toolbarItem.appendChild(toolbarIcon);
+        }
+
+        var toolbarBadge = document.createElement('div');
+        toolbarBadge.classList.add('jbadge');
+        var toolbarBadgeContent = document.createElement('div');
+        toolbarBadgeContent.innerHTML = options.items[i].badge ? options.items[i].badge : '';
+        toolbarBadge.appendChild(toolbarBadgeContent);
+        toolbarItem.appendChild(toolbarBadge);
+
+        if (options.items[i].title) {
+            var toolbarTitle = document.createElement('span');
+            toolbarTitle.innerHTML = options.items[i].title;
+            toolbarItem.appendChild(toolbarTitle);
+        }
+
+        toolbarContent.appendChild(toolbarItem);
+    }
+
+    el.toolbar = obj;
+
+    el.appendChild(toolbar);
+
+    return obj;
+});
+
+jSuites.actionsheet = (function() {
+    var actionsheet = document.createElement('div');
+    actionsheet.className = 'jactionsheet';
+    actionsheet.style.display = 'none';
+
+    var actionContent = document.createElement('div');
+    actionContent.className = 'jactionsheet-content';
+    actionsheet.appendChild(actionContent);
+
+    var obj = function(options) {
+        if (options) {
+            obj.options = options;
+        }
+
+        // Reset container
+        actionContent.innerHTML = '';
+
+        // Create new elements
+        for (var i = 0; i < obj.options.length; i++) {
+            var actionGroup = document.createElement('div');
+            actionGroup.className = 'jactionsheet-group';
+
+            for (var j = 0; j < obj.options[i].length; j++) {
+                var v = obj.options[i][j];
+                var actionItem = document.createElement('div');
+                var actionInput = document.createElement('input');
+                actionInput.type = 'button';
+                actionInput.value = v.title;
+                if (v.className) {
+                    actionInput.className = v.className; 
+                }
+                if (v.onclick) {
+                    actionInput.onclick = v.onclick; 
+                }
+                if (v.action == 'cancel') {
+                    actionInput.style.color = 'red';
+                }
+                actionItem.appendChild(actionInput);
+                actionGroup.appendChild(actionItem);
+            }
+
+            actionContent.appendChild(actionGroup);
+        }
+
+        // Show
+        actionsheet.style.display = '';
+
+        // Append
+        jSuites.el.appendChild(actionsheet);
+
+        // Animation
+        jSuites.slideBottom(actionContent, true);
+    }
+
+    obj.close = function() {
+        if (actionsheet.style.display != 'none') {
+            // Remove any existing actionsheet
+            jSuites.slideBottom(actionContent, false, function() {
+                actionsheet.remove();
+                actionsheet.style.display = 'none';
+            });
+        }
+    }
+
+    var mouseUp = function(e) {
+        obj.close();
+    }
+
+    actionsheet.addEventListener('mouseup', mouseUp);
+
+    obj.options = {};
+
+    return obj;
+})();
+
+/**
+ * (c) jSuites modal
+ * https://github.com/paulhodel/jsuites
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Modal
+ */
+
+jSuites.modal = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // Default configuration
+    var defaults = {
+        url: null,
+        onopen: null,
+        onclose: null,
+        closed: false,
+        width: null,
+        height: null,
+        title: null,
+    };
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Title
+    if (! obj.options.title && el.getAttribute('title')) {
+        obj.options.title = el.getAttribute('title');
+    }
+
+    var temp = document.createElement('div');
+    for (var i = 0; i < el.children.length; i++) {
+        temp.appendChild(el.children[i]);
+    }
+
+    obj.content = document.createElement('div');
+    obj.content.className = 'jmodal_content';
+    obj.content.innerHTML = el.innerHTML;
+
+    for (var i = 0; i < temp.children.length; i++) {
+        obj.content.appendChild(temp.children[i]);
+    }
+
+    obj.container = document.createElement('div');
+    obj.container.className = 'jmodal';
+    obj.container.appendChild(obj.content);
+
+    if (obj.options.width) {
+        obj.container.style.width = obj.options.width;
+    }
+    if (obj.options.height) {
+        obj.container.style.height = obj.options.height;
+    }
+    if (obj.options.title) {
+        obj.container.setAttribute('title', obj.options.title);
+    } else {
+        obj.container.classList.add('no-title');
+    }
+    el.innerHTML = '';
+    el.style.display = 'none';
+    el.appendChild(obj.container);
+
+    // Backdrop
+    var backdrop = document.createElement('div');
+    backdrop.className = 'jmodal_backdrop';
+    el.appendChild(backdrop);
+
+    obj.open = function() {
+        el.style.display = 'block';
+        // Fullscreen
+        const rect = obj.container.getBoundingClientRect();
+        if (jSuites.getWindowWidth() < rect.width) {
+            obj.container.style.top = '';
+            obj.container.style.left = '';
+            obj.container.classList.add('jmodal_fullscreen');
+            jSuites.slideBottom(obj.container, 1);
+        } else {
+            backdrop.style.display = 'block';
+        }
+        // Current
+        jSuites.modal.current = obj;
+        // Event
+        if (typeof(obj.options.onopen) == 'function') {
+            obj.options.onopen(el, obj);
+        }
+    }
+
+    obj.resetPosition = function() {
+        obj.container.style.top = '';
+        obj.container.style.left = '';
+    }
+
+    obj.isOpen = function() {
+        return el.style.display != 'none' ? true : false;
+    }
+
+    obj.close = function() {
+        el.style.display = 'none';
+        // Backdrop
+        backdrop.style.display = '';
+        // Current
+        jSuites.modal.current = null;
+        // Remove fullscreen class
+        obj.container.classList.remove('jmodal_fullscreen');
+        // Event
+        if (typeof(obj.options.onclose) == 'function') {
+            obj.options.onclose(el, obj);
+        }
+    }
+
+    if (! jSuites.modal.hasEvents) {
+        jSuites.modal.current = obj;
+
+        if ('ontouchstart' in document.documentElement === true) {
+            document.addEventListener("touchstart", jSuites.modal.mouseDownControls);
+        } else {
+            document.addEventListener('mousedown', jSuites.modal.mouseDownControls);
+            document.addEventListener('mousemove', jSuites.modal.mouseMoveControls);
+            document.addEventListener('mouseup', jSuites.modal.mouseUpControls);
+        }
+
+        document.addEventListener('keydown', jSuites.modal.keyDownControls);
+
+        jSuites.modal.hasEvents = true;
+    }
+
+    if (obj.options.url) {
+        jSuites.ajax({
+            url: obj.options.url,
+            method: 'GET',
+            success: function(data) {
+                obj.content.innerHTML = data;
+
+                if (! obj.options.closed) {
+                    obj.open();
+                }
+            }
+        });
+    } else {
+        if (! obj.options.closed) {
+            obj.open();
+        }
+    }
+
+    // Keep object available from the node
+    el.modal = obj;
+
+    return obj;
+});
+
+jSuites.modal.current = null;
+jSuites.modal.position = null;
+
+jSuites.modal.keyDownControls = function(e) {
+    if (e.which == 27) {
+        if (jSuites.modal.current) {
+            jSuites.modal.current.close();
+        }
+    }
+}
+
+jSuites.modal.mouseUpControls = function(e) {
+    if (jSuites.modal.current) {
+        jSuites.modal.current.container.style.cursor = 'auto';
+    }
+    jSuites.modal.position = null;
+}
+
+jSuites.modal.mouseMoveControls = function(e) {
+    if (jSuites.modal.current && jSuites.modal.position) {
+        if (e.which == 1 || e.which == 3) {
+            var position = jSuites.modal.position;
+            jSuites.modal.current.container.style.top = (position[1] + (e.clientY - position[3]) + (position[5] / 2)) + 'px';
+            jSuites.modal.current.container.style.left = (position[0] + (e.clientX - position[2]) + (position[4] / 2)) + 'px';
+            jSuites.modal.current.container.style.cursor = 'move';
+        } else {
+            jSuites.modal.current.container.style.cursor = 'auto';
+        }
+    }
+}
+
+jSuites.modal.mouseDownControls = function(e) {
+    jSuites.modal.position = [];
+
+    if (e.target.classList.contains('jmodal')) {
+        setTimeout(function() {
+            // Get target info
+            var rect = e.target.getBoundingClientRect();
+
+            if (e.changedTouches && e.changedTouches[0]) {
+                var x = e.changedTouches[0].clientX;
+                var y = e.changedTouches[0].clientY;
+            } else {
+                var x = e.clientX;
+                var y = e.clientY;
+            }
+
+            if (rect.width - (x - rect.left) < 50 && (y - rect.top) < 50) {
+                setTimeout(function() {
+                    jSuites.modal.current.close();
+                }, 100);
+            } else {
+                if (e.target.getAttribute('title') && (y - rect.top) < 50) {
+                    if (document.selection) {
+                        document.selection.empty();
+                    } else if ( window.getSelection ) {
+                        window.getSelection().removeAllRanges();
+                    }
+
+                    jSuites.modal.position = [
+                        rect.left,
+                        rect.top,
+                        e.clientX,
+                        e.clientY,
+                        rect.width,
+                        rect.height,
+                    ];
+                }
+            }
+        }, 100);
+    }
+}
+
+
+jSuites.notification = (function(options) {
+    var obj = {};
+    obj.options = {};
+
+    // Default configuration
+    var defaults = {
+        icon: null,
+        name: 'Notification',
+        date: null,
+        title: null,
+        message: null,
+        timeout: 4000,
+        autoHide: true,
+        closeable: true,
+    };
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    var notification = document.createElement('div');
+    notification.className = 'jnotification';
+
+    var notificationContainer = document.createElement('div');
+    notificationContainer.className = 'jnotification-container';
+    notification.appendChild(notificationContainer);
+
+    var notificationHeader = document.createElement('div');
+    notificationHeader.className = 'jnotification-header';
+    notificationContainer.appendChild(notificationHeader);
+
+    var notificationImage = document.createElement('div');
+    notificationImage.className = 'jnotification-image';
+    notificationHeader.appendChild(notificationImage);
+
+    if (obj.options.icon) {
+        var notificationIcon = document.createElement('img');
+        notificationIcon.src = obj.options.icon;
+        notificationImage.appendChild(notificationIcon);
+    }
+
+    var notificationName = document.createElement('div');
+    notificationName.className = 'jnotification-name';
+    notificationName.innerHTML = obj.options.name;
+    notificationHeader.appendChild(notificationName);
+
+    if (obj.options.closeable == true) {
+        var notificationClose = document.createElement('div');
+        notificationClose.className = 'jnotification-close';
+        notificationClose.onclick = function() {
+            obj.hide();
+        }
+        notificationHeader.appendChild(notificationClose);
+    }
+
+    var notificationDate = document.createElement('div');
+    notificationDate.className = 'jnotification-date';
+    notificationHeader.appendChild(notificationDate);
+
+    var notificationContent = document.createElement('div');
+    notificationContent.className = 'jnotification-content';
+    notificationContainer.appendChild(notificationContent);
+
+    if (obj.options.title) {
+        var notificationTitle = document.createElement('div');
+        notificationTitle.className = 'jnotification-title';
+        notificationTitle.innerHTML = obj.options.title;
+        notificationContent.appendChild(notificationTitle);
+    }
+
+    var notificationMessage = document.createElement('div');
+    notificationMessage.className = 'jnotification-message';
+    notificationMessage.innerHTML = obj.options.message;
+    notificationContent.appendChild(notificationMessage);
+
+    obj.show = function() {
+        document.body.appendChild(notification);
+        if (jSuites.getWindowWidth() > 800) { 
+            jSuites.fadeIn(notification);
+        } else {
+            jSuites.slideTop(notification, 1);
+        }
+    }
+
+    obj.hide = function() {
+        if (jSuites.getWindowWidth() > 800) { 
+            jSuites.fadeOut(notification, function() {
+                if (notification.parentNode) {
+                    notification.parentNode.removeChild(notification);
+                    if (notificationTimeout) {
+                        clearTimeout(notificationTimeout);
+                    }
+                }
+            });
+        } else {
+            jSuites.slideTop(notification, 0, function() {
+                if (notification.parentNode) {
+                    notification.parentNode.removeChild(notification);
+                    if (notificationTimeout) {
+                        clearTimeout(notificationTimeout);
+                    }
+                }
+            });
+        }
+    };
+
+    obj.show();
+
+    if (obj.options.autoHide == true) {
+        var notificationTimeout = setTimeout(function() {
+            obj.hide();
+        }, obj.options.timeout);
+    }
+
+    if (jSuites.getWindowWidth() < 800) {
+        notification.addEventListener("swipeup", function(e) {
+            obj.hide();
+            e.preventDefault();
+            e.stopPropagation();
+        });
+    }
+
+    return obj;
+});
+
+jSuites.rating = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // Default configuration
+    var defaults = {
+        number: 5,
+        value: 0,
+        tooltip: [ 'Very bad', 'Bad', 'Average', 'Good', 'Very good' ],
+        onchange: null,
+    };
+
+    // Loop through the initial configuration
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Class
+    el.classList.add('jrating');
+
+    // Add elements
+    for (var i = 0; i < obj.options.number; i++) {
+        var div = document.createElement('div');
+        div.setAttribute('data-index', (i + 1))
+        div.setAttribute('title', obj.options.tooltip[i])
+        el.appendChild(div);
+    }
+
+    // Set value
+    obj.setValue = function(index) {
+        for (var i = 0; i < obj.options.number; i++) {
+            if (i < index) {
+                el.children[i].classList.add('jrating-selected');
+            } else {
+                el.children[i].classList.remove('jrating-over');
+                el.children[i].classList.remove('jrating-selected');
+            }
+        }
+
+        obj.options.value = index;
+
+        if (typeof(obj.options.onchange) == 'function') {
+            obj.options.onchange(el, index);
+        }
+    }
+
+    obj.getValue = function() {
+        return obj.options.value;
+    }
+
+    if (obj.options.value) {
+        for (var i = 0; i < obj.options.number; i++) {
+            if (i < obj.options.value) {
+                el.children[i].classList.add('jrating-selected');
+            }
+        }
+    }
+
+    // Events
+    el.addEventListener("click", function(e) {
+        var index = e.target.getAttribute('data-index');
+        if (index != undefined) {
+            if (index == obj.options.value) {
+                obj.setValue(0);
+            } else {
+                obj.setValue(index);
+            }
+        }
+    });
+
+    el.addEventListener("mouseover", function(e) {
+        var index = e.target.getAttribute('data-index');
+        for (var i = 0; i < obj.options.number; i++) {
+            if (i < index) {
+                el.children[i].classList.add('jrating-over');
+            } else {
+                el.children[i].classList.remove('jrating-over');
+            }
+        }
+    });
+
+    el.addEventListener("mouseout", function(e) {
+        for (var i = 0; i < obj.options.number; i++) {
+            el.children[i].classList.remove('jrating-over');
+        }
+    });
+
+    el.rating = obj;
+
+    return obj;
+});
+
+
+/**
+ * (c) Image slider
+ * https://github.com/paulhodel/jtools
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Image Slider
+ */
+
+jSuites.slider = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+    obj.currentImage = null;
+
+    if (options) {
+        obj.options = options;
+    }
+
+    // Items
+    obj.options.items = [];
+
+    if (! el.classList.contains('jslider')) {
+        el.classList.add('jslider');
+
+        // Create container
+        var container = document.createElement('div');
+        container.className = 'jslider-container';
+
+        // Move children inside
+        if (el.children.length > 0) {
+            // Keep children items
+            for (var i = 0; i < el.children.length; i++) {
+                obj.options.items.push(el.children[i]);
+            }
+        }
+        if (obj.options.items.length > 0) {
+            for (var i = 0; i < obj.options.items.length; i++) {
+                obj.options.items[i].classList.add('jfile');
+                var index = obj.options.items[i].src.lastIndexOf('/');
+                if (index < 0) {
+                    obj.options.items[i].setAttribute('data-name', obj.options.items[i].src);
+                } else {
+                    obj.options.items[i].setAttribute('data-name', obj.options.items[i].src.substr(index + 1));
+                }
+                var index = obj.options.items[i].src.lastIndexOf('/');
+
+                container.appendChild(obj.options.items[i]);
+            }
+        }
+        el.appendChild(container);
+        // Add close buttom
+        var close = document.createElement('div');
+        close.className = 'jslider-close';
+        close.innerHTML = '';
+        close.onclick =  function() {
+            obj.close();
+        }
+        el.appendChild(close);
+    } else {
+        var container = el.querySelector('slider-container');
+    }
+
+    obj.show = function(target) {
+        if (! target) {
+            var target = container.children[0];
+        }
+
+        if (! container.classList.contains('jslider-preview')) {
+            container.classList.add('jslider-preview');
+            close.style.display = 'block';
+        }
+
+        // Hide all images
+        for (var i = 0; i < container.children.length; i++) {
+            container.children[i].style.display = 'none';
+        }
+
+        // Show clicked only
+        target.style.display = 'block';
+
+        // Is there any previous
+        if (target.previousSibling) {
+            container.classList.add('jslider-left');
+        } else {
+            container.classList.remove('jslider-left');
+        }
+
+        // Is there any next
+        if (target.nextSibling) {
+            container.classList.add('jslider-right');
+        } else {
+            container.classList.remove('jslider-right');
+        }
+
+        obj.currentImage = target;
+    }
+
+    obj.open = function() {
+        obj.show();
+
+        // Event
+        if (typeof(obj.options.onopen) == 'function') {
+            obj.options.onopen(el);
+        }
+    }
+
+    obj.close = function() {
+        container.classList.remove('jslider-preview');
+        container.classList.remove('jslider-left');
+        container.classList.remove('jslider-right');
+
+        for (var i = 0; i < container.children.length; i++) {
+            container.children[i].style.display = '';
+        }
+
+        close.style.display = '';
+
+        obj.currentImage = null;
+
+        // Event
+        if (typeof(obj.options.onclose) == 'function') {
+            obj.options.onclose(el);
+        }
+    }
+
+    obj.reset = function() {
+        container.innerHTML = '';
+    }
+
+    obj.addFile = function(v, ignoreEvents) {
+        var img = document.createElement('img');
+        img.setAttribute('data-lastmodified', v.lastmodified);
+        img.setAttribute('data-name', v.name);
+        img.setAttribute('data-size', v.size);
+        img.setAttribute('data-extension', v.extension);
+        img.setAttribute('data-cover', v.cover);
+        img.setAttribute('src', v.file);
+        img.className = 'jfile';
+        container.appendChild(img);
+        obj.options.items.push(img);
+
+        // Onchange
+        if (! ignoreEvents) {
+            if (typeof(obj.options.onchange) == 'function') {
+                obj.options.onchange(el, v);
+            }
+        }
+    }
+
+    obj.addFiles = function(files) {
+        for (var i = 0; i < files.length; i++) {
+            obj.addFile(files[i]);
+        }
+    }
+
+    obj.next = function() {
+        if (obj.currentImage.nextSibling) {
+            obj.show(obj.currentImage.nextSibling);
+        }
+    }
+    
+    obj.prev = function() {
+        if (obj.currentImage.previousSibling) {
+            obj.show(obj.currentImage.previousSibling);
+        }
+    }
+
+    obj.getData = function() {
+        return jSuites.getFiles(container);
+    }
+
+    // Append data
+    if (obj.options.data && obj.options.data.length) {
+        for (var i = 0; i < obj.options.data.length; i++) {
+            if (obj.options.data[i]) {
+                obj.addFile(obj.options.data[i]);
+            }
+        }
+    }
+
+    // Allow insert
+    if (obj.options.allowAttachment) {
+        var attachmentInput = document.createElement('input');
+        attachmentInput.type = 'file';
+        attachmentInput.className = 'slider-attachment';
+        attachmentInput.setAttribute('accept', 'image/*');
+        attachmentInput.style.display = 'none';
+        attachmentInput.onchange = function() {
+            var reader = [];
+
+            for (var i = 0; i < this.files.length; i++) {
+                var type = this.files[i].type.split('/');
+
+                if (type[0] == 'image') {
+                    var extension = this.files[i].name;
+                    extension = extension.split('.');
+                    extension = extension[extension.length-1];
+
+                    var file = {
+                        size: this.files[i].size,
+                        name: this.files[i].name,
+                        extension: extension,
+                        cover: 0,
+                        lastmodified: this.files[i].lastModified,
+                    }
+
+                    reader[i] = new FileReader();
+                    reader[i].addEventListener("load", function (e) {
+                        file.file = e.target.result;
+                        obj.addFile(file);
+                    }, false);
+
+                    reader[i].readAsDataURL(this.files[i]);
+                } else {
+                    alert('The extension is not allowed');
+                }
+            };
+        }
+
+        var attachmentIcon = document.createElement('i');
+        attachmentIcon.innerHTML = 'attachment';
+        attachmentIcon.className = 'jslider-attach material-icons';
+        attachmentIcon.onclick = function() {
+            jSuites.click(attachmentInput);
+        }
+
+        el.appendChild(attachmentInput);
+        el.appendChild(attachmentIcon);
+    }
+
+    // Push to refresh
+    var longTouchTimer = null;
+
+    var mouseDown = function(e) {
+        if (e.target.tagName == 'IMG') {
+            // Remove
+            var targetImage = e.target;
+            longTouchTimer = setTimeout(function() {
+                if (e.target.src.substr(0,4) == 'data') {
+                    e.target.remove();
+                } else {
+                    if (e.target.classList.contains('jremove')) {
+                        e.target.classList.remove('jremove');
+                    } else {
+                        e.target.classList.add('jremove');
+                    }
+                }
+
+                // Onchange
+                if (typeof(obj.options.onchange) == 'function') {
+                    obj.options.onchange(el, e.target);
+                }
+            }, 1000);
+        }
+    }
+
+    var mouseUp = function(e) {
+        if (longTouchTimer) {
+            clearTimeout(longTouchTimer);
+        }
+
+        // Open slider
+        if (e.target.tagName == 'IMG') {
+            if (! e.target.classList.contains('jremove')) {
+                obj.show(e.target);
+            }
+        } else {
+            // Arrow controls
+            if (e.target.clientWidth - e.offsetX < 40) {
+                // Show next image
+                obj.next();
+            } else if (e.offsetX < 40) {
+                // Show previous image
+                obj.prev();
+            }
+        }
+    }
+
+    container.addEventListener('mousedown', mouseDown);
+    container.addEventListener('touchstart', mouseDown);
+    container.addEventListener('mouseup', mouseUp);
+    container.addEventListener('touchend', mouseUp);
+
+    // Add global events
+    el.addEventListener("swipeleft", function(e) {
+        obj.next();
+        e.preventDefault();
+        e.stopPropagation();
+    });
+
+    el.addEventListener("swiperight", function(e) {
+        obj.prev();
+        e.preventDefault();
+        e.stopPropagation();
+    });
+
+
+    el.slider = obj;
+
+    return obj;
+});
+
+/**
+ * (c) jTools v1.0.1 - Element sorting
+ * https://github.com/paulhodel/jtools
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Element drag and drop sorting
+ */
+
+jSuites.sorting = (function(el, options) {
+    el.classList.add('jsorting');
+
+    el.addEventListener('dragstart', function(e) {
+        e.target.classList.add('jsorting_dragging');
+    });
+
+    el.addEventListener('dragover', function(e) {
+        e.preventDefault();
+
+        if (getElement(e.target) && ! e.target.classList.contains('jsorting')) {
+            if (e.target.clientHeight / 2 > e.offsetY) {
+                e.path[0].style.borderTop = '1px dotted red';
+                e.path[0].style.borderBottom = '';
+            } else {
+                e.path[0].style.borderTop = '';
+                e.path[0].style.borderBottom = '1px dotted red';
+            }
+        }
+    });
+
+    el.addEventListener('dragleave', function(e) {
+        e.path[0].style.borderTop = '';
+        e.path[0].style.borderBottom = '';
+    });
+
+    el.addEventListener('dragend', function(e) {
+        e.path[1].querySelector('.jsorting_dragging').classList.remove('jsorting_dragging');
+    });
+
+    el.addEventListener('drop', function(e) {
+        e.preventDefault();
+
+        if (getElement(e.target) && ! e.target.classList.contains('jsorting')) {
+            var element = e.path[1].querySelector('.jsorting_dragging');
+
+            if (e.target.clientHeight / 2 > e.offsetY) {
+                e.path[1].insertBefore(element, e.path[0]);
+            } else {
+                e.path[1].insertBefore(element, e.path[0].nextSibling);
+            }
+        }
+
+        e.path[0].style.borderTop = '';
+        e.path[0].style.borderBottom = '';
+    });
+
+    var getElement = function(element) {
+        var sorting = false;
+
+        function path (element) {
+            if (element.className) {
+                if (element.classList.contains('jsorting')) {
+                    sorting = true;
+                }
+            }
+
+            if (! sorting) {
+                path(element.parentNode);
+            }
+        }
+
+        path(element);
+
+        return sorting;
+    }
+
+    for (var i = 0; i < el.children.length; i++) {
+        el.children[i].setAttribute('draggable', 'true');
+    };
+
+    return el;
+});
+
+jSuites.tabs = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // Default configuration
+    var defaults = {
+        data: null,
+        allowCreate: false,
+        onload: null,
+        onchange: null,
+        oncreate: null,
+        animation: false,
+        create: null,
+        autoName: false,
+        prefixName: '',
+        hideHeaders: false,
+    };
+
+    // Loop through the initial configuration
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Class
+    el.classList.add('jtabs');
+
+    if (obj.options.animation == true) {
+        // Border
+        var border = document.createElement('div');
+        border.className = 'jtabs-border';
+        el.appendChild(border);
+
+        var setBorder = function(index) {
+            var rect = headers.children[index].getBoundingClientRect();
+            var rectContent = content.children[index].getBoundingClientRect();
+            border.style.width = rect.width + 'px';
+            border.style.left = (rect.left - rectContent.left) + 'px';
+            border.style.top = rect.height + 'px';
+        }
+    }
+
+    // Set value
+    obj.open = function(index) {
+        for (var i = 0; i < headers.children.length; i++) {
+            headers.children[i].classList.remove('jtabs-selected');
+            if (content.children[i]) {
+                content.children[i].classList.remove('jtabs-selected');
+            }
+        }
+
+        headers.children[index].classList.add('jtabs-selected');
+        if (content.children[index]) {
+            content.children[index].classList.add('jtabs-selected');
+        }
+
+        // Hide
+        if (obj.options.hideHeaders == true && (headers.children.length < 2 && obj.options.allowCreate == false)) {
+            headers.style.display = 'none';
+        } else {
+            headers.style.display = '';
+            // Set border
+            if (obj.options.animation == true) {
+                setTimeout(function() {
+                    setBorder(index);
+                }, 100);
+            }
+        }
+    }
+
+    obj.selectIndex = function(a) {
+        var index = Array.prototype.indexOf.call(headers.children, a);
+        if (index >= 0) {
+            obj.open(index);
+        }
+    }
+
+    obj.create = function(title, div) {
+        if (typeof(obj.options.create) == 'function') {
+            obj.options.create();
+        } else {
+            obj.appendElement(title, div);
+
+            if (typeof(obj.options.oncreate) == 'function') {
+                obj.options.oncreate(el, div)
+            }
+        }
+    }
+
+    obj.appendElement = function(title, div) {
+        if (! title) {
+            if (obj.options.autoName == true) {
+                var title = obj.options.prefixName;
+                title += ' ' + (parseInt(content.children.length) + 1);
+            } else {
+                var title = prompt('Title?', '');
+            }
+        }
+
+        if (title) {
+            // Add headers
+            var header = document.createElement('div');
+            header.innerHTML = title;
+            if (obj.options.allowCreate) {
+                headers.insertBefore(header, headers.lastChild);
+            } else {
+                headers.appendChild(header);
+            }
+
+            // Add content
+            if (! div) {
+                var div = document.createElement('div');
+            }
+            content.appendChild(div);
+
+            // Open new tab
+            obj.selectIndex(header);
+        }
+    }
+
+    // Create from data
+    if (obj.options.data) {
+        // Make sure the component is blank
+        el.innerHTML = '';
+        var headers = document.createElement('div');
+        var content = document.createElement('div');
+        headers.classList.add('jtabs-headers');
+        content.classList.add('jtabs-content');
+        el.appendChild(headers);
+        el.appendChild(content);
+
+        for (var i = 0; i < obj.options.data.length; i++) {
+            var headersItem = document.createElement('div');
+            headers.appendChild(headersItem);
+            var contentItem = document.createElement('div');
+            content.appendChild(contentItem);
+
+            headersItem.innerHTML = obj.options.data[i].title;
+            if (obj.options.data[i].content) {
+                contentItem.innerHTML = obj.options.data[i].content;
+            } else if (obj.options.data[i].url) {
+                jSuites.ajax({
+                    url: obj.options.data[i].url,
+                    type: 'GET',
+                    success: function(result) {
+                        contentItem.innerHTML = result;
+                    },
+                    complete: function() {
+                        if (typeof(obj.options.onload) == 'function') {
+                            obj.options.onload(el);
+
+                            obj.open(0);
+                        }
+                    }
+                });
+            }
+        }
+    } else if (el.children[0] && el.children[1]) {
+        // Create from existing elements
+        var headers = el.children[0];
+        var content = el.children[1];
+        headers.classList.add('jtabs-headers');
+        content.classList.add('jtabs-content');
+    } else {
+        el.innerHTML = '';
+        var headers = document.createElement('div');
+        var content = document.createElement('div');
+        headers.classList.add('jtabs-headers');
+        content.classList.add('jtabs-content');
+        el.appendChild(headers);
+        el.appendChild(content);
+    }
+
+    // New
+    if (obj.options.allowCreate == true) {
+        var add = document.createElement('i');
+        add.className = 'jtabs-add';
+        headers.appendChild(add);
+    }
+
+    // Events
+    headers.addEventListener("click", function(e) {
+        if (e.target.tagName == 'DIV') {
+            obj.selectIndex(e.target);
+        } else {
+            obj.create();
+        }
+    });
+
+    if (headers.children.length) {
+        obj.open(0);
+    }
+
+    el.tabs = obj;
+
+    return obj;
+});
+
+jSuites.tags = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    /**
+     * @typedef {Object} defaults
+     * @property {(string|Array)} value - Initial value of the compontent
+     * @property {number} limit - Max number of tags inside the element
+     * @property {string} search - The URL for suggestions
+     * @property {string} placeholder - The default instruction text on the element
+     * @property {validation} validation - Method to validate the tags
+     * @property {requestCallback} onbeforechange - Method to be execute before any changes on the element
+     * @property {requestCallback} onchange - Method to be execute after any changes on the element
+     * @property {requestCallback} onfocus - Method to be execute when on focus
+     * @property {requestCallback} onblur - Method to be execute when on blur
+     * @property {requestCallback} onload - Method to be execute when the element is loaded
+     */
+    var defaults = {
+        value: null,
+        limit: null,
+        limitMessage: 'The limit of entries is: ',
+        search: null,
+        placeholder: null,
+        validation: null,
+        onbeforechange: null,
+        onchange: null,
+        onfocus: null,
+        onblur: null,
+        onload: null,
+        colors: null,
+    };
+
+    // Loop through though the default configuration
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Search helpers
+    var searchContainer = null;
+    var searchTerms = null;
+    var searchIndex = 0;
+    var searchTimer = 0;
+
+    /**
+     * Add a new tag to the element
+     * @param {(?string|Array)} value - The value of the new element
+     */
+    obj.add = function(value, focus) {
+        if (typeof(obj.options.onbeforechange) == 'function') {
+            var v = obj.options.onbeforechange(el, obj, value);
+            if (v != null) {
+                value = v;
+            }
+        }
+
+        // Close search
+        if (searchContainer) {
+            searchContainer.style.display = '';
+        }
+
+        if (obj.options.limit > 0 && el.children.length >= obj.options.limit) {
+            alert(obj.options.limitMessage + ' ' + obj.options.limit);
+        } else {
+            // Get node
+            var node = getSelectionStart();
+
+            // Mix argument string or array
+            if (! value || typeof(value) == 'string') {
+                var div = document.createElement('div');
+                div.innerHTML = value ? value : '<br>';
+                if (node && node.parentNode.classList.contains('jtags')) {
+                    el.insertBefore(div, node.nextSibling);
+                } else {
+                    el.appendChild(div);
+                }
+            } else {
+                if (node && node.parentNode.classList.contains('jtags')) {
+                    if (! node.innerText.replace("\n", "")) {
+                        el.removeChild(node);
+                    }
+                }
+
+                for (var i = 0; i <= value.length; i++) {
+                    if (! obj.options.limit || el.children.length < obj.options.limit) {
+                        var div = document.createElement('div');
+                        div.innerHTML = value[i] ? value[i] : '<br>';
+                        el.appendChild(div);
+                    }
+                }
+            }
+
+            // Place caret
+            if (focus) {
+                setTimeout(function() {
+                    caret(div);
+                }, 0);
+            }
+
+            // Filter
+            filter();
+
+            if (typeof(obj.options.onchange) == 'function') {
+                obj.options.onchange(el, obj, value ? value : '');
+            }
+        }
+    }
+
+    obj.remove = function(node) {
+        // Remove node
+        node.parentNode.removeChild(node);
+        if (! el.children.length) {
+            obj.add('');
+        }
+    }
+
+    /**
+     * Get all tags in the element
+     * @return {Array} data - All tags as an array
+     */
+    obj.getData = function() {
+        var data = [];
+        for (var i = 0; i < el.children.length; i++) {
+            var value = obj.getValue(i);
+            if (value) {
+                var id = el.children[i].getAttribute('data-id');
+                if (! id) {
+                    id = value;
+                }
+                data.push({ id: id, value: value });
+            }
+        }
+        return data;
+    }
+
+    /**
+     * Get the value of one tag. Null for all tags
+     * @param {?number} index - Tag index number. Null for all tags.
+     * @return {string} value - All tags separated by comma
+     */
+    obj.getValue = function(index) {
+        var value = null;
+
+        if (index != null) {
+            // Get one individual value
+            value = el.children[index].innerText.replace("\n", "");
+        } else {
+            // Get all
+            var data = [];
+            for (var i = 0; i < el.children.length; i++) {
+                value = el.children[i].innerText.replace("\n", "");
+                if (value) {
+                    data.push(obj.getValue(i));
+                }
+            }
+            value = data.join(',');
+        }
+
+        return value;
+    }
+
+    /**
+     * Set the value of the element based on a string separeted by (,|;|\r\n)
+     * @param {string} value - A string with the tags
+     */
+    obj.setValue = function(text) {
+        // Remove whitespaces
+        text = text.trim();
+
+        if (text) {
+            // Tags
+            var data = extractTags(text);
+            // Add tags to the element
+            obj.add(data);
+        }
+    }
+
+    obj.reset = function() {
+        el.innerHTML = '<div><br></div>';
+    }
+
+    /**
+     * Verify if all tags in the element are valid
+     * @return {boolean}
+     */
+    obj.isValid = function() {
+        var test = 0;
+        for (var i = 0; i < el.children.length; i++) {
+            if (el.children[i].classList.contains('jtags_error')) {
+                test++;
+            }
+        }
+        return test == 0 ? true : false;
+    }
+
+    /**
+     * Add one element from the suggestions to the element
+     * @param {object} item - Node element in the suggestions container
+     */ 
+    obj.selectIndex = function(item) {
+        // Reset terms
+        searchTerms = '';
+        var node = getSelectionStart();
+        // Append text to the caret
+        node.innerText = item.children[1].innerText;
+        // Set node id
+        if (item.children[1].getAttribute('data-id')) {
+            node.setAttribute('data-id', item.children[1].getAttribute('data-id'));
+        }
+        // Close container
+        if (searchContainer) {
+            searchContainer.style.display = '';
+            searchContainer.innerHTML = '';
+        }
+        // Remove any error
+        node.classList.remove('jtags_error');
+        // Add new item
+        obj.add();
+    }
+
+    /**
+     * Search for suggestions
+     * @param {object} node - Target node for any suggestions
+     */
+    obj.search = function(node) {
+        // Create and append search container to the DOM
+        if (! searchContainer) {
+            var div = document.createElement('div');
+            div.style.position = 'relative';
+            el.parentNode.insertBefore(div, el.nextSibling);
+
+            // Create container
+            searchContainer = document.createElement('div');
+            searchContainer.classList.add('jtags_search');
+            div.appendChild(searchContainer);
+        }
+
+        // Search for
+        var terms = node.anchorNode.nodeValue;
+
+        // Search
+        if (node.anchorNode.nodeValue && terms != searchTerms) {
+            // Terms
+            searchTerms = node.anchorNode.nodeValue;
+            // Reset index
+            searchIndex = 0;
+            // Get remove results
+            jSuites.ajax({
+                url: obj.options.search + searchTerms,
+                method: 'GET',
+                dataType: 'json',
+                success: function(data) {
+                    // Reset container
+                    searchContainer.innerHTML = '';
+
+                    // Print results
+                    if (! data.length) {
+                        // Show container
+                        searchContainer.style.display = '';
+                    } else {
+                        // Show container
+                        searchContainer.style.display = 'block';
+
+                        // Show items
+                        var len = data.length < 11 ? data.length : 10;
+                        for (var i = 0; i < len; i++) {
+                            var div = document.createElement('div');
+                            if (i == 0) {
+                                div.classList.add('selected');
+                            }
+                            var img = document.createElement('img');
+                            if (data[i].image) {
+                                img.src = data[i].image;
+                            } else {
+                                img.style.display = 'none';
+                            }
+                            div.appendChild(img);
+
+                            var item = document.createElement('div');
+                            item.setAttribute('data-id', data[i].id);
+                            item.innerHTML = data[i].name;
+                            div.onclick = function() {
+                                // Add item
+                                obj.selectIndex(this);
+                            }
+                            div.appendChild(item);
+                            // Append item to the container
+                            searchContainer.appendChild(div);
+                        }
+                    }
+                }
+            });
+        }
+    }
+
+    // Destroy tags element
+    obj.destroy = function() {
+        // Bind events
+        el.removeEventListener('mouseup', tagsMouseUp);
+        el.removeEventListener('keydown', tagsKeyDown);
+        el.removeEventListener('keyup', tagsKeyUp);
+        el.removeEventListener('paste', tagsPaste);
+        el.removeEventListener('focus', tagsFocus);
+        el.removeEventListener('blur', tagsBlur);
+        // Remove element
+        el.parentNode.removeChild(el);
+    }
+
+    var getRandomColor = function(index) {
+        var rand = function(min, max) {
+            return min + Math.random() * (max - min);
+        }
+        return 'hsl(' + rand(1, 360) + ',' + rand(40, 70) + '%,' + rand(65, 72) + '%)';
+    }
+
+    /**
+     * Filter tags
+     */
+    var filter = function() {
+        for (var i = 0; i < el.children.length; i++) {
+            // Create label design
+            if (! obj.getValue(i)) {
+                el.children[i].classList.remove('jtags_label');
+            } else {
+                el.children[i].classList.add('jtags_label');
+
+                // Validation in place
+                if (typeof(obj.options.validation) == 'function') {
+                    if (obj.getValue(i)) {
+                        if (! obj.options.validation(el.children[i], el.children[i].innerText, el.children[i].getAttribute('data-id'))) {
+                            el.children[i].classList.add('jtags_error');
+                        } else {
+                            el.children[i].classList.remove('jtags_error');
+                        }
+                    } else {
+                        el.children[i].classList.remove('jtags_error');
+                    }
+                }
+            }
+        }
+    }
+
+    /**
+     * Place caret in the element node
+     */
+    var caret = function(e) {
+        var range = document.createRange();
+        var sel = window.getSelection();
+        range.setStart(e, e.innerText.length);
+        range.collapse(true);
+        sel.removeAllRanges();
+        sel.addRange(range);
+    }
+
+    /**
+     * Selection
+     */
+    var getSelectionStart = function() {
+        var node = document.getSelection().anchorNode;
+        if (node) {
+            return (node.nodeType == 3 ? node.parentNode : node);
+        } else {
+            return null;
+        }
+    }
+
+    /**
+     * Extract tags from a string
+     * @param {string} text - Raw string
+     * @return {Array} data - Array with extracted tags
+     */
+    var extractTags = function(text) {
+        /** @type {Array} */
+        var data = [];
+
+        /** @type {string} */
+        var word = '';
+
+        // Remove whitespaces
+        text = text.trim();
+
+        if (text) {
+            for (var i = 0; i < text.length; i++) {
+                if (text[i] == ',' || text[i] == ';' || text[i] == '\r\n') {
+                    if (word) {
+                        data.push(word);
+                        word = '';
+                    }
+                } else {
+                    word += text[i];
+                }
+            }
+
+            if (word) {
+                data.push(word);
+            }
+        }
+
+        return data;
+    }
+
+    /** @type {number} */
+    var anchorOffset = 0;
+
+    /**
+     * Processing event keydown on the element
+     * @param e {object}
+     */
+    var tagsKeyDown = function(e) {
+        // Anchoroffset
+        anchorOffset = window.getSelection().anchorOffset;
+
+        // Verify content
+        if (! el.children.length) {
+            var div = document.createElement('div');
+            div.innerHTML = '<br>';
+            el.appendChild(div);
+        }
+        // Comma
+        if (e.which == 9 || e.which == 186 || e.which == 188) {
+            var n = window.getSelection().anchorOffset;
+            if (n > 1) {
+                if (! obj.options.limit || el.children.length < obj.options.limit) {
+                    obj.add('', true);
+                }
+            }
+            e.preventDefault();
+        } else if (e.which == 13) {
+            // Enter
+            if (searchContainer && searchContainer.style.display != '') {
+                obj.selectIndex(searchContainer.children[searchIndex]);
+            } else {
+                var n = window.getSelection().anchorOffset;
+                if (n > 1) {
+                    if (! obj.options.limit || el.children.length < obj.options.limit) {
+                        obj.add('', true);
+                    }
+                }
+            }
+            e.preventDefault();
+        } else if (e.which == 38) {
+            // Up
+            if (searchContainer && searchContainer.style.display != '') {
+                searchContainer.children[searchIndex].classList.remove('selected');
+                if (searchIndex > 0) {
+                    searchIndex--;
+                }
+                searchContainer.children[searchIndex].classList.add('selected');
+                e.preventDefault();
+            }
+        } else if (e.which == 40) {
+            // Down
+            if (searchContainer && searchContainer.style.display != '') {
+                searchContainer.children[searchIndex].classList.remove('selected');
+                if (searchIndex < 9) {
+                    searchIndex++;
+                }
+                searchContainer.children[searchIndex].classList.add('selected');
+                e.preventDefault();
+            }
+        }
+    }
+
+    /**
+     * Processing event keyup on the element
+     * @param e {object}
+     */
+    var tagsKeyUp = function(e) {
+        if (e.which == 39) {
+            var n = window.getSelection().anchorOffset;
+            if (n > 1 && n == anchorOffset) {
+                obj.add();
+            }
+        } else if (e.which == 13 || e.which == 38 || e.which == 40) {
+            e.preventDefault();
+        } else {
+            if (searchTimer) {
+                clearTimeout(searchTimer);
+            }
+
+            searchTimer = setTimeout(function() {
+                // Current node
+                var node = window.getSelection();
+                // Search
+                if (obj.options.search) {
+                    obj.search(node);
+                }
+                searchTimer = null;
+            }, 300);
+        }
+
+        filter();
+    }
+
+    /**
+     * Processing event paste on the element
+     * @param e {object}
+     */
+    var tagsPaste =  function(e) {
+        if (e.clipboardData || e.originalEvent.clipboardData) {
+            var html = (e.originalEvent || e).clipboardData.getData('text/html');
+            var text = (e.originalEvent || e).clipboardData.getData('text/plain');
+        } else if (window.clipboardData) {
+            var html = window.clipboardData.getData('Html');
+            var text = window.clipboardData.getData('Text');
+        }
+
+        obj.setValue(text);
+        e.preventDefault();
+    }
+
+    /**
+     * Processing event mouseup on the element
+     * @param e {object}
+     */
+    var tagsMouseUp = function(e) {
+        if (e.target.parentNode && e.target.parentNode.classList.contains('jtags')) {
+            if (e.target.classList.contains('jtags_label') || e.target.classList.contains('jtags_error')) {
+                const rect = e.target.getBoundingClientRect();
+                if (rect.width - (e.clientX - rect.left) < 16) {
+                    obj.remove(e.target);
+                }
+            }
+        }
+
+        if (searchContainer) {
+            searchContainer.style.display = '';
+        }
+    }
+
+    /**
+     * Processing event focus on the element
+     * @param e {object}
+     */
+    var tagsFocus = function(e) {
+        if (! el.children.length || obj.getValue(el.children.length - 1)) {
+            if (! obj.options.limit || el.children.length < obj.options.limit) {
+                var div = document.createElement('div');
+                div.innerHTML = '<br>';
+                el.appendChild(div);
+            }
+        }
+
+        if (typeof(obj.options.onfocus) == 'function') {
+            obj.options.onfocus(el, obj, obj.getValue());
+        }
+    }
+
+    /**
+     * Processing event blur on the element
+     * @param e {object}
+     */
+    var tagsBlur = function(e) {
+        if (searchContainer) {
+            setTimeout(function() {
+                searchContainer.style.display = '';
+            }, 200);
+        }
+
+        for (var i = 0; i < el.children.length - 1; i++) {
+            // Create label design
+            if (! obj.getValue(i)) {
+                el.removeChild(el.children[i]);
+            }
+        }
+
+        if (typeof(obj.options.onblur) == 'function') {
+            obj.options.onblur(el, obj, obj.getValue());
+        }
+    }
+
+    // Bind events
+    el.addEventListener('mouseup', tagsMouseUp);
+    el.addEventListener('keydown', tagsKeyDown);
+    el.addEventListener('keyup', tagsKeyUp);
+    el.addEventListener('paste', tagsPaste);
+    el.addEventListener('focus', tagsFocus);
+    el.addEventListener('blur', tagsBlur);
+
+    // Prepare container
+    el.classList.add('jtags');
+    el.setAttribute('contenteditable', true);
+    el.setAttribute('spellcheck', false);
+
+    if (obj.options.placeholder) {
+        el.placeholder = obj.options.placeholder;
+    }
+
+    // Make sure element is empty
+    if (obj.options.value) {
+        obj.setValue(obj.options.value);
+    } else {
+        el.innerHTML = '<div><br></div>';
+    }
+
+    if (typeof(obj.options.onload) == 'function') {
+        obj.options.onload(el, obj);
+    }
+
+    el.tags = obj;
+
+    return obj;
+});
+
+/**
+ * (c) jSuites template renderer
+ * https://github.com/paulhodel/jsuites
+ *
+ * @author: Paul Hodel <paul.hodel@gmail.com>
+ * @description: Template renderer
+ */
+
+jSuites.template = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // Default configuration
+    var defaults = {
+        url: null,
+        data: null,
+        filter: null,
+        pageNumber: 0,
+        numberOfPages: 0,
+        template: null,
+        render: null,
+        onload: null,
+        onchange: null,
+        noRecordsFound: 'No records found',
+        // Searchable
+        search: null,
+        searchInput: true,
+        // Search happens in the back end
+        searchValue: '',
+        remoteData: null,
+        // Pagination page number of items
+        pagination: null,
+    }
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    // Reset content
+    el.innerHTML = '';
+
+    // Input search
+    if (obj.options.search && obj.options.searchInput == true) {
+        // Timer
+        var searchTimer = null;
+
+        // Search container
+        var searchContainer = document.createElement('div');
+        searchContainer.className = 'jtemplate-results';
+        var searchInput = document.createElement('input');
+        searchInput.value = obj.options.searchValue;
+        searchInput.onkeyup = function(e) {
+            // Clear current trigger
+            if (searchTimer) {
+                clearTimeout(searchTimer);
+            }
+            // Prepare search
+            searchTimer = setTimeout(function() {
+                obj.search(searchInput.value.toLowerCase());
+                searchTimer = null;
+            }, 300)
+        }
+        searchContainer.appendChild(searchInput);
+        el.appendChild(searchContainer);
+    }
+
+    // Pagination container
+    if (obj.options.pagination) {
+        var pagination = document.createElement('div');
+        pagination.className = 'jtemplate-pagination';
+        el.appendChild(pagination);
+    }
+
+    // Content
+    var container = document.createElement('div');
+    container.className = 'jtemplate-content';
+    el.appendChild(container);
+
+    // Data container
+    var searchResults = null;
+
+    obj.updatePagination = function() {
+        // Reset pagination container
+        if (pagination) {
+            pagination.innerHTML = '';
+        }
+
+        // Create pagination
+        if (obj.options.pagination > 0 && obj.options.numberOfPages > 1) {
+            // Number of pages
+            var numberOfPages = obj.options.numberOfPages;
+
+            // Controllers
+            if (obj.options.pageNumber < 6) {
+                var startNumber = 1;
+                var finalNumber = numberOfPages < 10 ? numberOfPages : 10;
+            } else if (numberOfPages - obj.options.pageNumber < 5) {
+                var startNumber = numberOfPages - 9;
+                var finalNumber = numberOfPages;
+                if (startNumber < 1) {
+                    startNumber = 1;
+                }
+            } else {
+                var startNumber = obj.options.pageNumber - 4;
+                var finalNumber = obj.options.pageNumber + 5;
+            }
+
+            // First
+            if (startNumber > 1) {
+                var paginationItem = document.createElement('div');
+                paginationItem.innerHTML = '<';
+                paginationItem.title = 1;
+                pagination.appendChild(paginationItem);
+            }
+
+            // Get page links
+            for (var i = startNumber; i <= finalNumber; i++) {
+                var paginationItem = document.createElement('div');
+                paginationItem.innerHTML = i;
+                pagination.appendChild(paginationItem);
+
+                if (obj.options.pageNumber == i - 1) {
+                    paginationItem.style.fontWeight = 'bold';
+                }
+            }
+
+            // Last
+            if (finalNumber < numberOfPages) {
+                var paginationItem = document.createElement('div');
+                paginationItem.innerHTML = '>';
+                paginationItem.title = numberOfPages - 1;
+                pagination.appendChild(paginationItem);
+            }
+        }
+    }
+
+    obj.addItem = function(data, beginOfDataSet) {
+        // Append itens
+        var content = document.createElement('div');
+        // Append data
+        if (beginOfDataSet) {
+            obj.options.data.unshift(data);
+        } else {
+            obj.options.data.push(data);
+        }
+        // Get content
+        content.innerHTML = obj.options.template[Object.keys(options.template)[0]](data);
+        // Add animation
+        content.children[0].classList.add('fade-in');
+        // Add and do the animation
+        if (beginOfDataSet) {
+            container.prepend(content.children[0]);
+        } else {
+            container.append(content.children[0]);
+        }
+        // Onchange method
+        if (typeof(obj.options.onchange) == 'function') {
+            obj.options.onchange(el, obj.options.data);
+        }
+    }
+
+    obj.removeItem = function(element) {
+        if (Array.prototype.indexOf.call(container.children, element) > -1) {
+            // Remove data from array
+            var index = obj.options.data.indexOf(element.dataReference);
+            if (index > -1) {
+                obj.options.data.splice(index, 1);
+            }
+            // Remove element from DOM
+            jSuites.fadeOut(element, function() {
+                element.parentNode.removeChild(element);
+            });
+        } else {
+            console.error('Element not found');
+        }
+    }
+
+    obj.setData = function(data) {
+        if (data) {
+            obj.options.pageNumber = 1;
+            obj.options.searchValue = '';
+            // Set data
+            obj.options.data = data;
+            // Reset any search results
+            searchResults = null;
+
+            // Render new data
+            obj.render();
+
+            // Onchange method
+            if (typeof(obj.options.onchange) == 'function') {
+                obj.options.onchange(el, obj.options.data);
+            }
+        }
+    }
+
+    obj.appendData = function(data, pageNumber) {
+        if (pageNumber) {
+            obj.options.pageNumber = pageNumber;
+        }
+
+        var execute = function(data) {
+            // Concat data
+            obj.options.data.concat(data);
+            // Number of pages
+            if (obj.options.pagination > 0) {
+                obj.options.numberOfPages = Math.ceil(obj.options.data.length / obj.options.pagination);
+            }
+            var startNumber = 0;
+            var finalNumber = data.length;
+            // Append itens
+            var content = document.createElement('div');
+            for (var i = startNumber; i < finalNumber; i++) {
+                content.innerHTML = obj.options.template[Object.keys(obj.options.template)[0]](data[i]);
+                content.children[0].dataReference = data[i]; 
+                container.appendChild(content.children[0]);
+            }
+        }
+
+        if (obj.options.url && obj.options.remoteData == true) {
+            // URL
+            var url = obj.options.url;
+            // Data
+            var ajaxData = {};
+            // Options for backend search
+            if (obj.options.remoteData) {
+                // Search value
+                if (obj.options.searchValue) {
+                    ajaxData.q = obj.options.searchValue;
+                }
+                // Page number
+                if (obj.options.pageNumber) {
+                    ajaxData.p = obj.options.pageNumber;
+                }
+                // Number items per page
+                if (obj.options.pagination) {
+                    ajaxData.t = obj.options.pagination;
+                }
+            }
+            // Remote loading
+            jSuites.ajax({
+                url: url,
+                method: 'GET',
+                data: ajaxData,
+                dataType: 'json',
+                success: function(data) {
+                    execute(data);
+                }
+            });
+        } else {
+            if (! obj.options.data) {
+                console.log('TEMPLATE: no data or external url defined');
+            } else {
+                execute(data);
+            }
+        }
+    }
+
+    obj.renderTemplate = function() {
+        // Data container
+        var data = searchResults ? searchResults : obj.options.data;
+
+        // Reset pagination
+        obj.updatePagination();
+
+        if (! data.length) {
+            container.innerHTML = obj.options.noRecordsFound;
+            container.classList.add('jtemplate-empty');
+        } else {
+            // Reset content
+            container.classList.remove('jtemplate-empty');
+
+            // Create pagination
+            if (obj.options.pagination && data.length > obj.options.pagination) {
+                var startNumber = (obj.options.pagination * obj.options.pageNumber);
+                var finalNumber = (obj.options.pagination * obj.options.pageNumber) + obj.options.pagination;
+
+                if (data.length < finalNumber) {
+                    var finalNumber = data.length;
+                }
+            } else {
+                var startNumber = 0;
+                var finalNumber = data.length;
+            }
+
+            // Append itens
+            var content = document.createElement('div');
+            for (var i = startNumber; i < finalNumber; i++) {
+                content.innerHTML = obj.options.template[Object.keys(obj.options.template)[0]](data[i]);
+                content.children[0].dataReference = data[i]; 
+                container.appendChild(content.children[0]);
+            }
+        }
+    }
+
+    obj.render = function(pageNumber, forceLoad) {
+        // Update page number
+        if (pageNumber != undefined) {
+            obj.options.pageNumber = pageNumber;
+        } else {
+            if (! obj.options.pageNumber && obj.options.pagination > 0) {
+                obj.options.pageNumber = 0;
+            }
+        }
+
+        // Render data into template
+        var execute = function() {
+            // Render new content
+            if (typeof(obj.options.render) == 'function') {
+                container.innerHTML = obj.options.render(obj);
+            } else {
+                container.innerHTML = '';
+            }
+
+            // Load data
+            obj.renderTemplate();
+
+            // Onload
+            if (forceLoad && typeof(obj.options.onload) == 'function') {
+                obj.options.onload(el, obj);
+            }
+        }
+
+        if (obj.options.url && (obj.options.remoteData == true || forceLoad)) {
+            // URL
+            var url = obj.options.url;
+            // Data
+            var ajaxData = {};
+            // Options for backend search
+            if (obj.options.remoteData) {
+                // Search value
+                if (obj.options.searchValue) {
+                    ajaxData.q = obj.options.searchValue;
+                }
+                // Page number
+                if (obj.options.pageNumber) {
+                    ajaxData.p = obj.options.pageNumber;
+                }
+                // Number items per page
+                if (obj.options.pagination) {
+                    ajaxData.t = obj.options.pagination;
+                }
+            }
+            // Remote loading
+            jSuites.ajax({
+                url: url,
+                method: 'GET',
+                dataType: 'json',
+                data: ajaxData,
+                success: function(data) {
+                    // Search and keep data in the client side
+                    if (data.hasOwnProperty("total")) {
+                        obj.options.numberOfPages = Math.ceil(data.total / obj.options.pagination);
+                        obj.options.data = data.data;
+                    } else {
+                        // Number of pages
+                        if (obj.options.pagination > 0) {
+                            obj.options.numberOfPages = Math.ceil(data.length / obj.options.pagination);
+                        }
+                        obj.options.data = data;
+                    }
+
+                    // Load data for the user
+                    execute();
+                }
+            });
+        } else {
+            if (! obj.options.data) {
+                console.log('TEMPLATE: no data or external url defined');
+            } else {
+                // Number of pages
+                if (obj.options.pagination > 0) {
+                    if (searchResults) {
+                        obj.options.numberOfPages = Math.ceil(searchResults.length / obj.options.pagination);
+                    } else {
+                        obj.options.numberOfPages = Math.ceil(obj.options.data.length / obj.options.pagination);
+                    }
+                }
+                // Load data for the user
+                execute();
+            }
+        }
+    }
+
+    obj.search = function(query) {
+        obj.options.pageNumber = 0;
+        obj.options.searchValue = query ? query : '';
+
+        // Filter data
+        if (obj.options.remoteData == true || ! query) {
+            searchResults = null;
+        } else {
+            var test = function(o, query) {
+                for (var key in o) {
+                    var value = o[key];
+
+                    if ((''+value).toLowerCase().search(query) >= 0) {
+                        return true;
+                    }
+                }
+                return false;
+            }
+
+            if (typeof(obj.options.filter) == 'function') {
+                searchResults = obj.options.filter(obj.options.data, query);
+            } else {
+                searchResults = obj.options.data.filter(function(item) {
+                    return test(item, query);
+                });
+            }
+        }
+
+        obj.render();
+    }
+
+    obj.refresh = function() {
+        obj.render();
+    }
+
+    obj.reload = function() {
+        obj.render(0, true);
+    }
+
+    el.addEventListener('mousedown', function(e) {
+        if (e.target.parentNode.classList.contains('jtemplate-pagination')) {
+            var index = e.target.innerText;
+            if (index == '<') {
+                obj.render(0);
+            } else if (index == '>') {
+                obj.render(e.target.getAttribute('title'));
+            } else {
+                obj.render(parseInt(index)-1);
+            }
+            e.preventDefault();
+        }
+    });
+
+    el.template = obj;
+
+    // Render data
+    obj.render(0, true);
+
+    return obj;
+});
+
+jSuites.tracker = (function(el, options) {
+    var obj = {};
+    obj.options = {};
+
+    // Default configuration
+    var defaults = {
+        url: null,
+        message: 'Are you sure? There are unsaved information in your form',
+        ignore: false,
+        currentHash: null,
+        submitButton:null,
+        onload: null,
+        onbeforesave: null,
+        onsave: null,
+    };
+
+    // Loop through our object
+    for (var property in defaults) {
+        if (options && options.hasOwnProperty(property)) {
+            obj.options[property] = options[property];
+        } else {
+            obj.options[property] = defaults[property];
+        }
+    }
+
+    obj.setUrl = function(url) {
+        obj.options.url = url;
+    }
+
+    obj.load = function() {
+        jSuites.ajax({
+            url: obj.options.url,
+            method: 'GET',
+            dataType: 'json',
+            success: function(data) {
+                var elements = el.querySelectorAll("input, select, textarea");
+ 
+                for (var i = 0; i < elements.length; i++) {
+                    var name = elements[i].getAttribute('name');
+                    if (data[name]) {
+                        elements[i].value = data[name];
+                    }
+                }
+
+                if (typeof(obj.options.onload) == 'function') {
+                    obj.options.onload(el, data);
+                }
+            }
+        });
+    }
+
+    obj.save = function() {
+        var test = obj.validate();
+
+        if (test) {
+            jSuites.alert(test);
+        } else {
+            var data = obj.getElements(true);
+
+            if (typeof(obj.options.onbeforesave) == 'function') {
+                var data = obj.options.onbeforesave(el, data);
+
+                if (data === false) {
+                    console.log('Onbeforesave returned false');
+                    return; 
+                }
+            }
+
+            jSuites.ajax({
+                url: obj.options.url,
+                method: 'POST',
+                dataType: 'json',
+                data: data,
+                success: function(result) {
+                    jSuites.alert(result.message);
+
+                    if (typeof(obj.options.onsave) == 'function') {
+                        var data = obj.options.onsave(el, result);
+                    }
+
+                    obj.reset();
+                }
+            });
+        }
+    }
+
+    obj.validateElement = function(element) {
+        var emailChecker = function(data) {
+            var pattern = new RegExp(/^([\w-\.]+@([\w-]+\.)+[\w-]{2,4})?$/);
+            return pattern.test(data) ? true : false; 
+        }
+
+        var passwordChecker = function(data) {
+            return (data.length > 5) ? true : false;
+        }
+
+        var addError = function(element) {
+            // Add error in the element
+            element.classList.add('error');
+            // Submit button
+            if (obj.options.submitButton) {
+                obj.options.submitButton.setAttribute('disabled', true);
+            }
+            // Return error message
+            return element.getAttribute('data-error') || 'There is an error in the form';
+        }
+
+        var delError = function(element) {
+            var error = false;
+            // Remove class from this element
+            element.classList.remove('error');
+            // Get elements in the form
+            var elements = el.querySelectorAll("input, select, textarea");
+            // Run all elements 
+            for (var i = 0; i < elements.length; i++) {
+                if (elements[i].getAttribute('data-validation')) {
+                    if (elements[i].classList.contains('error')) {
+                        error = true;
+                    }
+                }
+            }
+
+            if (obj.options.submitButton) {
+                if (error) {
+                    obj.options.submitButton.setAttribute('disabled', true);
+                } else {
+                    obj.options.submitButton.removeAttribute('disabled');
+                }
+            }
+        }
+
+        // Blank
+        var test = '';
+        if (! element.value) {
+            test = addError(element);
+        } else if (element.getAttribute('data-email') && ! emailChecker(element.value)) {
+            test = addError(element);
+        } else if (element.getAttribute('data-password') && ! emailChecker(element.value)) {
+            test = addError(element);
+        } else {
+            if (element.classList.contains('error')) {
+                delError(element);
+            }
+        }
+
+        return test;
+    }
+
+    // Run form validation
+    obj.validate = function() {
+        var test = '';
+        // Get elements in the form
+        var elements = el.querySelectorAll("input, select, textarea");
+        // Run all elements 
+        for (var i = 0; i < elements.length; i++) {
+            if (elements[i].getAttribute('data-validation')) {
+                if (test) {
+                    test += "<br>\r\n";
+                }
+                test += obj.validateElement(elements[i]);
+            }
+        }
+        return test;
+    }
+
+    // Check the form
+    obj.getError = function() {
+        // Validation
+        return obj.validation() ? true : false;
+    }
+
+    // Return the form hash
+    obj.setHash = function() {
+        return obj.getHash(obj.getElements());
+    }
+
+    // Get the form hash
+    obj.getHash = function(str) {
+        var hash = 0, i, chr;
+
+        if (str.length === 0) {
+            return hash;
+        } else {
+            for (i = 0; i < str.length; i++) {
+              chr = str.charCodeAt(i);
+              hash = ((hash << 5) - hash) + chr;
+              hash |= 0;
+            }
+        }
+
+        return hash;
+    }
+
+    // Is there any change in the form since start tracking?
+    obj.isChanged = function() {
+        var hash = obj.setHash();
+        return (obj.options.currentHash != hash);
+    }
+
+    // Restart tracking
+    obj.resetTracker = function() {
+        obj.options.currentHash = obj.setHash();
+        obj.options.ignore = false;
+    }
+
+    obj.reset = function() {
+        obj.options.currentHash = obj.setHash();
+        obj.options.ignore = false;
+    }
+
+    // Ignore flag
+    obj.setIgnore = function(ignoreFlag) {
+        obj.options.ignore = ignoreFlag ? true : false;
+    }
+
+    // Get form elements
+    obj.getElements = function(asArray) {
+        var data = {};
+        var elements = el.querySelectorAll("input, select, textarea");
+
+        for (var i = 0; i < elements.length; i++) {
+            var element = elements[i];
+            var name = element.name;
+            var value = element.value;
+
+            if (name) {
+                data[name] = value;
+            }
+        }
+
+        return asArray == true ? data : JSON.stringify(data);
+    }
+
+    // Start tracking in one second
+    setTimeout(function() {
+        obj.options.currentHash = obj.setHash();
+    }, 1000);
+
+    // Alert
+    window.addEventListener("beforeunload", function (e) {
+        if (obj.isChanged() && obj.options.ignore == false) {
+            var confirmationMessage =  obj.options.message? obj.options.message : "\o/";
+
+            if (confirmationMessage) {
+                if (typeof e == 'undefined') {
+                    e = window.event;
+                }
+
+                if (e) {
+                    e.returnValue = confirmationMessage;
+                }
+
+                return confirmationMessage;
+            } else {
+                return void(0);
+            }
+        }
+    });
+
+    // Validations
+    el.addEventListener("keyup", function(e) {
+        if (e.target.getAttribute('data-validation')) {
+            obj.validateElement(e.target);
+        }
+    });
+
+    el.tracker = obj;
+
+    return obj;
+});
+
+
+
+    return jSuites;
+
+})));
\ No newline at end of file
diff --git a/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.js.xml b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.js.xml
new file mode 100644
index 0000000000000000000000000000000000000000..3726e4bcff0dbdfde0966f059795d3c115c7157f
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/SkinTemplateItem/portal_skins/erp5_jexcel_editor/jsuites/jsuites.js.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>jsuites.js</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>application/x-javascript</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_jexcel_editor/bt/description b/bt5/erp5_jexcel_editor/bt/description
new file mode 100644
index 0000000000000000000000000000000000000000..43fcf17e8e882d7686973d222685e49b595b906b
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/bt/description
@@ -0,0 +1 @@
+Spreadsheet editor with Jexcel library.
\ No newline at end of file
diff --git a/bt5/erp5_jexcel_editor/bt/template_format_version b/bt5/erp5_jexcel_editor/bt/template_format_version
new file mode 100644
index 0000000000000000000000000000000000000000..56a6051ca2b02b04ef92d5150c9ef600403cb1de
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/bt/template_format_version
@@ -0,0 +1 @@
+1
\ No newline at end of file
diff --git a/bt5/erp5_jexcel_editor/bt/template_skin_id_list b/bt5/erp5_jexcel_editor/bt/template_skin_id_list
new file mode 100644
index 0000000000000000000000000000000000000000..4d91c4f6c7ef494d7e263e9bb60d488a54f23f1a
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/bt/template_skin_id_list
@@ -0,0 +1 @@
+erp5_jexcel_editor
\ No newline at end of file
diff --git a/bt5/erp5_jexcel_editor/bt/title b/bt5/erp5_jexcel_editor/bt/title
new file mode 100644
index 0000000000000000000000000000000000000000..4d91c4f6c7ef494d7e263e9bb60d488a54f23f1a
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/bt/title
@@ -0,0 +1 @@
+erp5_jexcel_editor
\ No newline at end of file
diff --git a/bt5/erp5_jexcel_editor/bt/version b/bt5/erp5_jexcel_editor/bt/version
new file mode 100644
index 0000000000000000000000000000000000000000..0f3016682b38e6ad179e8bafb3008b4cde4251e7
--- /dev/null
+++ b/bt5/erp5_jexcel_editor/bt/version
@@ -0,0 +1 @@
+001
\ No newline at end of file
diff --git a/bt5/erp5_officejs_jquery_app/PathTemplateItem/portal_types/Web%20Table/web_table_clone.xml b/bt5/erp5_officejs_jquery_app/PathTemplateItem/portal_types/Web%20Table/web_table_clone.xml
new file mode 100644
index 0000000000000000000000000000000000000000..707b39d604aa295b3c4daacdab5f78e37015d500
--- /dev/null
+++ b/bt5/erp5_officejs_jquery_app/PathTemplateItem/portal_types/Web%20Table/web_table_clone.xml
@@ -0,0 +1,75 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="Action Information" module="erp5.portal_type"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>action</string> </key>
+            <value>
+              <persistent> <string encoding="base64">AAAAAAAAAAI=</string> </persistent>
+            </value>
+        </item>
+        <item>
+            <key> <string>action_permission</string> </key>
+            <value>
+              <tuple>
+                <string>Add portal content</string>
+              </tuple>
+            </value>
+        </item>
+        <item>
+            <key> <string>categories</string> </key>
+            <value>
+              <tuple>
+                <string>action_type/object_jio_js_script</string>
+              </tuple>
+            </value>
+        </item>
+        <item>
+            <key> <string>description</string> </key>
+            <value>
+              <none/>
+            </value>
+        </item>
+        <item>
+            <key> <string>id</string> </key>
+            <value> <string>web_table_clone</string> </value>
+        </item>
+        <item>
+            <key> <string>language</string> </key>
+            <value>
+              <none/>
+            </value>
+        </item>
+        <item>
+            <key> <string>portal_type</string> </key>
+            <value> <string>Action Information</string> </value>
+        </item>
+        <item>
+            <key> <string>reference</string> </key>
+            <value> <string>web_table_clone</string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string>Clone</string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+  <record id="2" aka="AAAAAAAAAAI=">
+    <pickle>
+      <global name="Expression" module="Products.CMFCore.Expression"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>text</string> </key>
+            <value> <string>string:${object_url}/Base_cloneDocumentForWebTable</string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_officejs_web_table_configuration.xml b/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_officejs_web_table_configuration.xml
index 3905b41a22444f5ad2a36179e9316215bdb2252b..99c1b3faca8f4be1056b31be675144ded11bc53e 100644
--- a/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_officejs_web_table_configuration.xml
+++ b/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_officejs_web_table_configuration.xml
@@ -110,7 +110,7 @@
         <item>
             <key> <string>text_content</string> </key>
             <value> <string>CONFIGURATION MANIFEST\n
-# generated on Fri Dec 13 20:31:13 2019\n
+# generated on Tue Jul 28 08:56:19 2020\n
 CACHE:\n
 \n
 hateoas_appcache/definition_view/cG9ydGFsX3R5cGVzL1dlYiBQYWdlIE1vZHVsZQ==\n
@@ -119,6 +119,8 @@ hateoas_appcache/definition_view/cG9ydGFsX3NraW5zL2VycDVfd2ViX3RhYmxlX2VkaXRvci9
 hateoas_appcache/definition_view/cG9ydGFsX3R5cGVzL1dlYiBUYWJsZQ==\n
 hateoas_appcache/definition_view/cG9ydGFsX3R5cGVzL1dlYiBUYWJsZS93ZWJfdGFibGVfdmlldw==\n
 hateoas_appcache/definition_view/cG9ydGFsX3NraW5zL2VycDVfd2ViX3RhYmxlX2VkaXRvci9XZWJUYWJsZV92aWV3QXNKaW9Gb3JXZWJUYWJsZUVkaXRvcg==\n
+hateoas_appcache/definition_view/cG9ydGFsX3R5cGVzL1dlYiBUYWJsZS93ZWJfdGFibGVfY2xvbmU=\n
+hateoas_appcache/definition_view/cG9ydGFsX3NraW5zL2VycDVfd2ViX3RhYmxlX2VkaXRvci9CYXNlX2Nsb25lRG9jdW1lbnRGb3JXZWJUYWJsZQ==\n
 \n
 NETWORK:\n
 *</string> </value>
diff --git a/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_web_table_router.html.html b/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_web_table_router.html.html
index 05cec78f5ddfe80e26f4ce6fad482cc6c060f071..2984d4aa4b5d0dc41441fe45dcfa3b80b341f275 100644
--- a/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_web_table_router.html.html
+++ b/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_web_table_router.html.html
@@ -19,7 +19,7 @@
     <script data-renderjs-configuration="parent_relative_url" type="text/x-renderjs-configuration">web_page_module</script>
     <script data-renderjs-configuration="parent_portal_type" type="text/x-renderjs-configuration">Web Page Module</script>
     <script data-renderjs-configuration="portal_skin_folder" type="text/x-renderjs-configuration">erp5_web_table_editor</script>
-    <script data-renderjs-configuration="app_actions" type="text/x-renderjs-configuration">('Web Table | web_table_view', 'Web Page Module | web_table_view')</script>
+    <script data-renderjs-configuration="app_actions" type="text/x-renderjs-configuration">('Web Table | web_table_view', 'Web Page Module | web_table_view', 'Web Table | web_table_clone')</script>
     <script data-renderjs-configuration="app_allowed_sub_types" type="text/x-renderjs-configuration">('Web Page Module | Web Table',)</script>
     <script data-renderjs-configuration="app_view_reference" type="text/x-renderjs-configuration">web_table_view</script>
     <script data-renderjs-configuration="web_page_module_dict" type="text/x-renderjs-configuration">{"front_page": 1, "editable": 0, "hide_add_button": 0, "jump_button": 0, "fast_input_button": 0, "export_button": 0, "filter_action": 1, "panel_action": 1, "previous_next_button": 0, "history_previous_link": 0, "title": "Jquery Sheet Documents", "hide_listbox_buttons": 1, "blob_type": "", "blob_create_object_url": 0}</script>
diff --git a/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_web_table_router.html.xml b/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_web_table_router.html.xml
index 914738b554cbaee5cc822859e5ec892363c47456..96d1fde0f3a6ea9a2ee31c616d924b95c29f10bf 100644
--- a/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_web_table_router.html.xml
+++ b/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_page_module/gadget_web_table_router.html.xml
@@ -235,7 +235,7 @@
                   </item>
                   <item>
                       <key> <string>serial</string> </key>
-                      <value> <string>977.62260.6779.17339</string> </value>
+                      <value> <string>984.59943.25702.48622</string> </value>
                   </item>
                   <item>
                       <key> <string>state</string> </key>
@@ -253,7 +253,7 @@
                           </tuple>
                           <state>
                             <tuple>
-                              <float>1568973158.24</float>
+                              <float>1595923550.24</float>
                               <string>UTC</string>
                             </tuple>
                           </state>
diff --git a/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_site_module/officejs_web_table_editor/app.xml b/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_site_module/officejs_web_table_editor/app.xml
index f1e272a070da2d0fc1337c52454e65e605536e7b..73193b0cfb18aac461ae8bdf73617adeb25a8741 100644
--- a/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_site_module/officejs_web_table_editor/app.xml
+++ b/bt5/erp5_officejs_jquery_app/PathTemplateItem/web_site_module/officejs_web_table_editor/app.xml
@@ -412,7 +412,8 @@
         <item>
             <key> <string>configuration_precache_manifest_script_list</string> </key>
             <value> <string>WebSection_getOfficeJsPrecacheManifestList\n
-WebSection_getWebTableEditorPrecacheManifestList</string> </value>
+WebSection_getWebTableEditorPrecacheManifestList\n
+WebSection_getJexcelEditorPrecacheManifestList</string> </value>
         </item>
         <item>
             <key> <string>configuration_router_gadget_url</string> </key>
@@ -677,7 +678,7 @@ WebSection_getWebTableEditorPrecacheManifestList</string> </value>
                   </item>
                   <item>
                       <key> <string>serial</string> </key>
-                      <value> <string>983.23018.43853.17715</string> </value>
+                      <value> <string>984.59943.25702.48622</string> </value>
                   </item>
                   <item>
                       <key> <string>state</string> </key>
@@ -695,7 +696,7 @@ WebSection_getWebTableEditorPrecacheManifestList</string> </value>
                           </tuple>
                           <state>
                             <tuple>
-                              <float>1587729708.72</float>
+                              <float>1594816416.45</float>
                               <string>UTC</string>
                             </tuple>
                           </state>
diff --git a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/Base_cloneDocumentForWebTable.xml b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/Base_cloneDocumentForWebTable.xml
new file mode 100644
index 0000000000000000000000000000000000000000..e360dfdbb86758ce07fc16544a60fa28a4408f2c
--- /dev/null
+++ b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/Base_cloneDocumentForWebTable.xml
@@ -0,0 +1,116 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="ERP5 Form" module="erp5.portal_type"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>_objects</string> </key>
+            <value>
+              <tuple/>
+            </value>
+        </item>
+        <item>
+            <key> <string>action</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>encoding</string> </key>
+            <value> <string>UTF-8</string> </value>
+        </item>
+        <item>
+            <key> <string>enctype</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>group_list</string> </key>
+            <value>
+              <list>
+                <string>left</string>
+                <string>right</string>
+                <string>center</string>
+                <string>bottom</string>
+                <string>hidden</string>
+              </list>
+            </value>
+        </item>
+        <item>
+            <key> <string>groups</string> </key>
+            <value>
+              <dictionary>
+                <item>
+                    <key> <string>bottom</string> </key>
+                    <value>
+                      <list/>
+                    </value>
+                </item>
+                <item>
+                    <key> <string>center</string> </key>
+                    <value>
+                      <list/>
+                    </value>
+                </item>
+                <item>
+                    <key> <string>hidden</string> </key>
+                    <value>
+                      <list/>
+                    </value>
+                </item>
+                <item>
+                    <key> <string>left</string> </key>
+                    <value>
+                      <list>
+                        <string>action_description</string>
+                        <string>gadget_field_action_js_script</string>
+                      </list>
+                    </value>
+                </item>
+                <item>
+                    <key> <string>right</string> </key>
+                    <value>
+                      <list/>
+                    </value>
+                </item>
+              </dictionary>
+            </value>
+        </item>
+        <item>
+            <key> <string>id</string> </key>
+            <value> <string>Base_cloneDocumentForWebTable</string> </value>
+        </item>
+        <item>
+            <key> <string>method</string> </key>
+            <value> <string>POST</string> </value>
+        </item>
+        <item>
+            <key> <string>name</string> </key>
+            <value> <string>Base_cloneDocumentForJexcelEditor</string> </value>
+        </item>
+        <item>
+            <key> <string>portal_type</string> </key>
+            <value> <string>ERP5 Form</string> </value>
+        </item>
+        <item>
+            <key> <string>row_length</string> </key>
+            <value> <int>4</int> </value>
+        </item>
+        <item>
+            <key> <string>stored_encoding</string> </key>
+            <value> <string>UTF-8</string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value>
+              <none/>
+            </value>
+        </item>
+        <item>
+            <key> <string>unicode_mode</string> </key>
+            <value> <int>0</int> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/Base_cloneDocumentForWebTable/action_description.xml b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/Base_cloneDocumentForWebTable/action_description.xml
new file mode 100644
index 0000000000000000000000000000000000000000..475edfbcad226a1ada84e08d6208c468cf00da35
--- /dev/null
+++ b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/Base_cloneDocumentForWebTable/action_description.xml
@@ -0,0 +1,139 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="LabelField" module="Products.Formulator.StandardFields"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>id</string> </key>
+            <value> <string>action_description</string> </value>
+        </item>
+        <item>
+            <key> <string>message_values</string> </key>
+            <value>
+              <dictionary/>
+            </value>
+        </item>
+        <item>
+            <key> <string>overrides</string> </key>
+            <value>
+              <dictionary>
+                <item>
+                    <key> <string>css_class</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>default</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>description</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>editable</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>enabled</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>extra</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>hidden</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>title</string> </key>
+                    <value> <string></string> </value>
+                </item>
+              </dictionary>
+            </value>
+        </item>
+        <item>
+            <key> <string>tales</string> </key>
+            <value>
+              <dictionary>
+                <item>
+                    <key> <string>css_class</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>default</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>description</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>editable</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>enabled</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>extra</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>hidden</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>title</string> </key>
+                    <value> <string></string> </value>
+                </item>
+              </dictionary>
+            </value>
+        </item>
+        <item>
+            <key> <string>values</string> </key>
+            <value>
+              <dictionary>
+                <item>
+                    <key> <string>css_class</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>default</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>description</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>editable</string> </key>
+                    <value> <int>1</int> </value>
+                </item>
+                <item>
+                    <key> <string>enabled</string> </key>
+                    <value> <int>1</int> </value>
+                </item>
+                <item>
+                    <key> <string>extra</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>hidden</string> </key>
+                    <value> <int>0</int> </value>
+                </item>
+                <item>
+                    <key> <string>title</string> </key>
+                    <value> <string>Confirm clone</string> </value>
+                </item>
+              </dictionary>
+            </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/Base_cloneDocumentForWebTable/gadget_field_action_js_script.xml b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/Base_cloneDocumentForWebTable/gadget_field_action_js_script.xml
new file mode 100644
index 0000000000000000000000000000000000000000..55111cd50cb7f2690e455a9b694588436a1cc26f
--- /dev/null
+++ b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/Base_cloneDocumentForWebTable/gadget_field_action_js_script.xml
@@ -0,0 +1,246 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="GadgetField" module="Products.ERP5Form.GadgetField"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>id</string> </key>
+            <value> <string>gadget_field_action_js_script</string> </value>
+        </item>
+        <item>
+            <key> <string>message_values</string> </key>
+            <value>
+              <dictionary>
+                <item>
+                    <key> <string>external_validator_failed</string> </key>
+                    <value> <string>The input failed the external validator.</string> </value>
+                </item>
+                <item>
+                    <key> <string>no_validator</string> </key>
+                    <value> <string>Does not support this operation.</string> </value>
+                </item>
+              </dictionary>
+            </value>
+        </item>
+        <item>
+            <key> <string>overrides</string> </key>
+            <value>
+              <dictionary>
+                <item>
+                    <key> <string>alternate_name</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>css_class</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>data_url</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>default</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>description</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>editable</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>enabled</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>external_validator</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>extra</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>gadget_url</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>hidden</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>js_sandbox</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>renderjs_extra</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>title</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>validator_field_id</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>validator_form_id</string> </key>
+                    <value> <string></string> </value>
+                </item>
+              </dictionary>
+            </value>
+        </item>
+        <item>
+            <key> <string>tales</string> </key>
+            <value>
+              <dictionary>
+                <item>
+                    <key> <string>alternate_name</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>css_class</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>data_url</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>default</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>description</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>editable</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>enabled</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>external_validator</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>extra</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>gadget_url</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>hidden</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>js_sandbox</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>renderjs_extra</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>title</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>validator_field_id</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>validator_form_id</string> </key>
+                    <value> <string></string> </value>
+                </item>
+              </dictionary>
+            </value>
+        </item>
+        <item>
+            <key> <string>values</string> </key>
+            <value>
+              <dictionary>
+                <item>
+                    <key> <string>alternate_name</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>css_class</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>data_url</string> </key>
+                    <value> <int>0</int> </value>
+                </item>
+                <item>
+                    <key> <string>default</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>description</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>editable</string> </key>
+                    <value> <int>1</int> </value>
+                </item>
+                <item>
+                    <key> <string>enabled</string> </key>
+                    <value> <int>1</int> </value>
+                </item>
+                <item>
+                    <key> <string>external_validator</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>extra</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>gadget_url</string> </key>
+                    <value> <string>action_web_table_clone.html</string> </value>
+                </item>
+                <item>
+                    <key> <string>hidden</string> </key>
+                    <value> <int>0</int> </value>
+                </item>
+                <item>
+                    <key> <string>js_sandbox</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>renderjs_extra</string> </key>
+                    <value>
+                      <list/>
+                    </value>
+                </item>
+                <item>
+                    <key> <string>title</string> </key>
+                    <value> <string>gadget_field_action_js_script</string> </value>
+                </item>
+                <item>
+                    <key> <string>validator_field_id</string> </key>
+                    <value> <string></string> </value>
+                </item>
+                <item>
+                    <key> <string>validator_form_id</string> </key>
+                    <value> <string></string> </value>
+                </item>
+              </dictionary>
+            </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/WebSection_getWebTableEditorPrecacheManifestList.py b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/WebSection_getWebTableEditorPrecacheManifestList.py
index 69ec684d269ecd0f8a7adb5b3c9ef1787a56ccfe..4c99f1920760df6fb8dfcd9bdf470f94f0f4cb34 100644
--- a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/WebSection_getWebTableEditorPrecacheManifestList.py
+++ b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/WebSection_getWebTableEditorPrecacheManifestList.py
@@ -4,43 +4,10 @@ url_list = [
   "gadget_officejs_web_table_router.html",
   "gadget_officejs_jio_web_table_view.html",
   "gadget_officejs_jio_web_table_view.js",
+  "action_web_table_clone.html",
+  "action_web_table_clone.js",
   "gadget_officejs_web_table.json",
   "web_table_app_logo.svg",
-
-  #jquery-sheets
-  "jquery.js",
-  "jquery-sheets.gadget.html",
-  "jquery-sheets.gadget.js",
-  "jquery-sheets/wickedgrid.css",
-  "jquery-sheets/menu.html",
-  "jquery-sheets/font-awesome.css",
-  "jquery-sheets/js/thaw.js",
-  "jquery-sheets/js/raphael.js",
-  "jquery-sheets/js/copy3_of_undomanager.js",
-  "jquery-sheets/js/jquery-ui.js",
-  "jquery-sheets/js/globalize.js",
-  "jquery-sheets/js/jquery.nearest.js",
-  "jquery-sheets/js/megatable.js",
-  "jquery-sheets/js/wickedgrid.js",
-  "jquery-sheets/js/graphael/g.bar.js",
-  "jquery-sheets/js/graphael/g.line.js",
-  "jquery-sheets/js/graphael/g.raphael.js",
-  "jquery-sheets/js/graphael/g.pie.js",
-  "jquery-sheets/js/graphael/g.dot.js",
-  "jquery-sheets/js/ZeroClipboard.js",
-  "jquery-sheets/js/jquery.elastic.source.js",
-  "jquery-sheets/js/infiniscroll.js",
-  "jquery-sheets/js/undomanager.js",
-  "jquery-sheets/js/MouseWheel.js",
-  "jquery-sheets/bootstrap-theme.css",
-  "jquery-sheets/images/underline1.svg",
-  "jquery-sheets/images/strikethrough.svg",
-  "jquery-sheets/images/right183.svg",
-  "jquery-sheets/images/left181.svg",
-  "jquery-sheets/images/centered5.svg",
-  "jquery-sheets/images/italic3.svg",
-  "jquery-sheets/images/bold13.svg",
-  "jquery-sheets/bootstrap.css",
 ]
 
 return url_list
diff --git a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/WebTable_viewAsJioForWebTableEditor/my_text_content.xml b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/WebTable_viewAsJioForWebTableEditor/my_text_content.xml
index 3e28e523fca09b0fac687014efdf9a5474b6affe..ea911048bac80125393165492575fd44ab9ed21f 100644
--- a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/WebTable_viewAsJioForWebTableEditor/my_text_content.xml
+++ b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/WebTable_viewAsJioForWebTableEditor/my_text_content.xml
@@ -224,8 +224,8 @@
                     <value>
                       <list>
                         <tuple>
-                          <string>{"editor": "jquery-sheets", "maximize": true}</string>
-                          <string>{"editor": "jquery-sheets", "maximize": true}</string>
+                          <string>{"editor": "jexcel", "maximize": true}</string>
+                          <string>{"editor": "jexcel", "maximize": true}</string>
                         </tuple>
                       </list>
                     </value>
diff --git a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.html.html b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.html.html
new file mode 100644
index 0000000000000000000000000000000000000000..479d48572950fd8a841edf959f93f2ca2e9c4f78
--- /dev/null
+++ b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.html.html
@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<html>
+  <head>
+    <meta charset="utf-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1">
+    <title>Web Table Clone</title>
+
+    <script src="rsvp.js"></script>
+    <script src="renderjs.js"></script>
+
+    <script src="action_web_table_clone.js"></script>
+  </head>
+  <body>
+  </body>
+</html>
\ No newline at end of file
diff --git a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.html.xml b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.html.xml
new file mode 100644
index 0000000000000000000000000000000000000000..9731023dde16ecdc7936802f3db82a5358aae890
--- /dev/null
+++ b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.html.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>action_web_table_clone.html</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>text/html</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.js.js b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.js.js
new file mode 100644
index 0000000000000000000000000000000000000000..c52506100fa4cb68ac533c1cb80d5de8faafd3f1
--- /dev/null
+++ b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.js.js
@@ -0,0 +1,57 @@
+/*jslint nomen: true, indent: 2 */
+/*global window, rJS, RSVP*/
+(function (window, rJS) {
+  "use strict";
+  rJS(window)
+    .declareAcquiredMethod("jio_get", "jio_get")
+    .declareAcquiredMethod("jio_post", "jio_post")
+    .declareMethod("preRenderDocument", function (parent_options) {
+      var gadget = this;
+      return gadget.jio_get(parent_options.jio_key)
+        .push(function (parent_document) {
+          return parent_document;
+        });
+    })
+    .declareMethod("handleSubmit", function (content_dict, parent_options) {
+      //must return a dict with:
+      //notify: options_dict for notifySubmitted
+      //redirect: options_dict for redirect
+      var return_submit_dict = {
+        notify: {
+          message: "",
+          status: ""
+        },
+        redirect: {
+          command: "display",
+          options: {}
+        }
+      },
+        gadget = this,
+        document = parent_options.doc,
+        property;
+      delete content_dict.dialog_method;
+      for (property in content_dict) {
+        if (content_dict.hasOwnProperty(property)) {
+          document[property] = content_dict[property];
+        }
+      }
+      return gadget.jio_post(document)
+        .push(function (jio_key) {
+          return_submit_dict.notify.message = "Document Cloned";
+          return_submit_dict.notify.status = "success";
+          return_submit_dict.redirect.options = {
+            jio_key: jio_key,
+            editable: true
+          };
+          return return_submit_dict;
+        }, function (error) {
+          if (error instanceof jIO.util.jIOError) {
+            return_submit_dict.notify.message = "Failure cloning document";
+            return_submit_dict.notify.status = "error";
+            return return_submit_dict;
+          }
+          throw error;
+        });
+    });
+
+}(window, rJS));
\ No newline at end of file
diff --git a/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.js.xml b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.js.xml
new file mode 100644
index 0000000000000000000000000000000000000000..d4345103f0720a404339ba04276f2dfb5a138efa
--- /dev/null
+++ b/bt5/erp5_officejs_jquery_app/SkinTemplateItem/portal_skins/erp5_web_table_editor/action_web_table_clone.js.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="File" module="OFS.Image"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>__name__</string> </key>
+            <value> <string>action_web_table_clone.js</string> </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>application/javascript</string> </value>
+        </item>
+        <item>
+            <key> <string>precondition</string> </key>
+            <value> <string></string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <string></string> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_officejs_jquery_app/bt/dependency_list b/bt5/erp5_officejs_jquery_app/bt/dependency_list
index 3306c17352a13dd90bf69d2d63933ca39939b768..4b8bc687f7f9411a2a740ec0bb2b3f531d6acc0b 100644
--- a/bt5/erp5_officejs_jquery_app/bt/dependency_list
+++ b/bt5/erp5_officejs_jquery_app/bt/dependency_list
@@ -1,3 +1,4 @@
 erp5_officejs
 erp5_jquery_sheet_js_editor
-erp5_minipaint
\ No newline at end of file
+erp5_minipaint
+erp5_jexcel_editor
\ No newline at end of file
diff --git a/bt5/erp5_officejs_jquery_app/bt/template_path_list b/bt5/erp5_officejs_jquery_app/bt/template_path_list
index e060ebe3fc2be74a47f1d4cd75b50fc65ee4845d..9717521187341df53456fec122dcb3436914e601 100644
--- a/bt5/erp5_officejs_jquery_app/bt/template_path_list
+++ b/bt5/erp5_officejs_jquery_app/bt/template_path_list
@@ -3,6 +3,7 @@ image_module/web_table_app_logo_svg
 portal_types/Image Module/image_editor_view
 portal_types/Image/image_editor_view
 portal_types/Web Page Module/web_table_view
+portal_types/Web Table/web_table_clone
 portal_types/Web Table/web_table_view
 web_page_module/gadget_image_editor*
 web_page_module/gadget_officejs_image_editor*
@@ -11,4 +12,4 @@ web_page_module/gadget_web_table*
 web_site_module/officejs_image_editor
 web_site_module/officejs_image_editor/**
 web_site_module/officejs_web_table_editor
-web_site_module/officejs_web_table_editor/**
+web_site_module/officejs_web_table_editor/**
\ No newline at end of file
diff --git a/bt5/erp5_officejs_ui_test/PathTemplateItem/portal_tests/officejs_ui_webtable_zuite/testJexcelEditor.xml b/bt5/erp5_officejs_ui_test/PathTemplateItem/portal_tests/officejs_ui_webtable_zuite/testJexcelEditor.xml
new file mode 100644
index 0000000000000000000000000000000000000000..593fb50b1624384479aa10a933fafeac4aaa2193
--- /dev/null
+++ b/bt5/erp5_officejs_ui_test/PathTemplateItem/portal_tests/officejs_ui_webtable_zuite/testJexcelEditor.xml
@@ -0,0 +1,58 @@
+<?xml version="1.0"?>
+<ZopeData>
+  <record id="1" aka="AAAAAAAAAAE=">
+    <pickle>
+      <global name="ZopePageTemplate" module="Products.PageTemplates.ZopePageTemplate"/>
+    </pickle>
+    <pickle>
+      <dictionary>
+        <item>
+            <key> <string>_bind_names</string> </key>
+            <value>
+              <object>
+                <klass>
+                  <global name="NameAssignments" module="Shared.DC.Scripts.Bindings"/>
+                </klass>
+                <tuple/>
+                <state>
+                  <dictionary>
+                    <item>
+                        <key> <string>_asgns</string> </key>
+                        <value>
+                          <dictionary>
+                            <item>
+                                <key> <string>name_subpath</string> </key>
+                                <value> <string>traverse_subpath</string> </value>
+                            </item>
+                          </dictionary>
+                        </value>
+                    </item>
+                  </dictionary>
+                </state>
+              </object>
+            </value>
+        </item>
+        <item>
+            <key> <string>content_type</string> </key>
+            <value> <string>text/html</string> </value>
+        </item>
+        <item>
+            <key> <string>expand</string> </key>
+            <value> <int>0</int> </value>
+        </item>
+        <item>
+            <key> <string>id</string> </key>
+            <value> <string>testJexcelEditor</string> </value>
+        </item>
+        <item>
+            <key> <string>output_encoding</string> </key>
+            <value> <string>utf-8</string> </value>
+        </item>
+        <item>
+            <key> <string>title</string> </key>
+            <value> <unicode></unicode> </value>
+        </item>
+      </dictionary>
+    </pickle>
+  </record>
+</ZopeData>
diff --git a/bt5/erp5_officejs_ui_test/PathTemplateItem/portal_tests/officejs_ui_webtable_zuite/testJexcelEditor.zpt b/bt5/erp5_officejs_ui_test/PathTemplateItem/portal_tests/officejs_ui_webtable_zuite/testJexcelEditor.zpt
new file mode 100644
index 0000000000000000000000000000000000000000..0b7be1b6995a52e5d02a484fc27384205623617b
--- /dev/null
+++ b/bt5/erp5_officejs_ui_test/PathTemplateItem/portal_tests/officejs_ui_webtable_zuite/testJexcelEditor.zpt
@@ -0,0 +1,177 @@
+<html xmlns:tal="http://xml.zope.org/namespaces/tal"
+      xmlns:metal="http://xml.zope.org/namespaces/metal">
+<head>
+<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
+<title>Test OfficeJS UI</title>
+</head>
+<body>
+<table cellpadding="1" cellspacing="1" border="1">
+<thead>
+<tr><td rowspan="1" colspan="3">Test OfficeJS Jexcel UI</td></tr>
+</thead><tbody>
+<tal:block metal:use-macro="here/Zuite_CommonTemplate/macros/init" />
+<!-- Clean Up -->
+<tr>
+  <td>open</td>
+  <td>${base_url}/bar_module/ListBoxZuite_reset</td>
+  <td></td>
+</tr>
+<tr>
+  <td>assertTextPresent</td>
+  <td>Reset Successfully.</td>
+  <td></td>
+</tr>
+<!-- Initialize and skip app installation -->
+<tr>
+  <td>open</td>
+  <td>${base_url}/web_site_module/officejs_web_table_editor/</td>
+  <td></td>
+</tr>
+
+<tr>
+  <td>waitForElementPresent</td>
+  <td>link=Skip</td>
+  <td></td>
+</tr>
+<tr>
+  <td>click</td>
+  <td>link=Skip</td>
+  <td></td>
+</tr>
+
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//a[@data-i18n='Storages']</td>
+  <td></td>
+</tr>
+<tr>
+  <td>click</td>
+  <td>//a[@data-i18n='Storages']</td>
+  <td></td>
+</tr>
+<tr>
+  <td>waitForElementPresent</td>
+  <td>link=Local is Enough</td>
+  <td></td>
+</tr>
+<tr>
+  <td>click</td>
+  <td>link=Local is Enough</td>
+  <td></td>
+</tr>
+<tr>
+  <td>waitForElementPresent</td>
+  <td>link=Add</td>
+  <td></td>
+</tr>
+<tr>
+  <td>click</td>
+  <td>link=Add</td>
+  <td></td>
+</tr>
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//button[@data-i18n='Save']</td>
+  <td></td>
+</tr>
+<tr>
+<!-- Wait for spreadsheet to be rendered by the gadget -->
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//div[@class='spreadsheet']</td>
+  <td></td>
+</tr>
+<!-- Verify maximize button is present -->
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//button[contains(@class, 'ui-icon-expand ui-btn-icon-notext')]</td>
+  <td></td>
+</tr>
+<!-- Verify editor has been maximized -->
+<tr>
+  <td>click</td>
+  <td>//button[contains(@class, 'ui-icon-expand ui-btn-icon-notext')]</td>
+  <td></td>
+</tr>
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//div[contains(@class, 'editor-maximize')]</td>
+  <td></td>
+</tr>
+<!-- Verify searchbar is present -->
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//div[contains(@class, 'jexcel_filter')]</td>
+  <td></td>
+</tr>
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//div[contains(@class, 'jexcel_filter')]</td>
+  <td></td>
+</tr>
+<!-- Verify toolbar is present -->
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//div[contains(@class, 'jexcel_toolbar')]</td>
+  <td></td>
+</tr>
+<!-- Verify body is present, draggable and resizable -->
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//tbody[contains(@class, 'draggable resizable')]</td>
+  <td></td>
+</tr>
+<!-- Add table -->
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//i[@title='Add table']</td>
+  <td></td>
+</tr>
+<tr>
+  <td>click</td>
+  <td>//i[@title='Add table']</td>
+  <td></td>
+</tr>
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//div[@data-spreadsheet='1']</td>
+  <td></td>
+</tr>
+<tr>
+  <td>assertText</td>
+  <td>//div[@data-spreadsheet='1']</td>
+  <td>Table 2</td>
+</tr>
+<!-- Delete Sheet -->
+<tr>
+  <td>click</td>
+  <td>//div[@data-spreadsheet='0']</td>
+  <td></td>
+</tr>
+<tr>
+  <td>waitForElementPresent</td>
+  <td>//i[@title='Delete table']</td>
+  <td></td>
+</tr>
+<tr>
+  <td>click</td>
+  <td>//i[@title='Delete table']</td>
+  <td></td>
+</tr>
+<tr>
+  <td>assertConfirmation</td>
+  <td>Delete this table ?</td>
+  <td></td>
+</tr>
+<tr>
+  <td>assertElementNotPresent</td>
+  <td>//div[@data-spreadsheet='1']</td>
+  <td></td>
+</tr>
+<tr>
+  <td>assertText</td>
+  <td>//div[@data-spreadsheet='0']</td>
+  <td>Table 1</td>
+</tr>
+</tbody>
+</table>
\ No newline at end of file
diff --git a/bt5/erp5_officejs_ui_test/PathTemplateItem/portal_tests/officejs_ui_webtable_zuite/testOfficeJSWebTableEditor.zpt b/bt5/erp5_officejs_ui_test/PathTemplateItem/portal_tests/officejs_ui_webtable_zuite/testOfficeJSWebTableEditor.zpt
index 1901f8a68d612bdd17aa445766dbfd51f3064cd3..0caf76de0125bcf3e1d9ee65df2506428d1bc178 100644
--- a/bt5/erp5_officejs_ui_test/PathTemplateItem/portal_tests/officejs_ui_webtable_zuite/testOfficeJSWebTableEditor.zpt
+++ b/bt5/erp5_officejs_ui_test/PathTemplateItem/portal_tests/officejs_ui_webtable_zuite/testOfficeJSWebTableEditor.zpt
@@ -116,7 +116,7 @@
 </tr>
 <tr>
   <td>waitForElementPresent</td>
-  <td>identifier=jquery_sheet_gadget</td>
+  <td>//div[@class='spreadsheet']</td>
   <td></td>
 </tr>
 <!-- Wait for editor completly loaded, this is workaround, wait for lock in renderjs -->
diff --git a/product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/gadget_editor.js.js b/product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/gadget_editor.js.js
index f63db3a8234f53852d751d39784a39d45b109b08..fcef0cc6ae306caf44b4b2e02c230ecf6d33be3e 100644
--- a/product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/gadget_editor.js.js
+++ b/product/ERP5/bootstrap/erp5_core/SkinTemplateItem/portal_skins/erp5_core/gadget_editor.js.js
@@ -13,7 +13,8 @@
     "jquery-sheets": {"url": "jquery-sheets.gadget.html"},
     "pdf": {"url": "pdf_js/pdfjs.gadget.html"},
     "notebook_editor": {"url": "gadget_notebook.html"},
-    "jsmd_editor": {"url": "gadget_jsmd_viewer.html"}
+    "jsmd_editor": {"url": "gadget_jsmd_viewer.html"},
+    "jexcel" : {"url": "jexcel.gadget.html"}
   };