diff --git a/app/assets/javascripts/.licensee.json b/app/assets/javascripts/.licensee.json index ae2abbd2b64..2a3a6cc6801 100644 --- a/app/assets/javascripts/.licensee.json +++ b/app/assets/javascripts/.licensee.json @@ -22,7 +22,9 @@ "messageformat": "0.1.5", "regenerator-transform": "0.10.1", "source-map": "0.1.43", - "sourcemap-validator": "1.1.1" + "sourcemap-validator": "1.1.1", + "jspreadsheet-ce": "4.13.4", + "@jspreadsheet/formula": "2.0.2" }, "corrections": true, "ignore": [ diff --git a/app/assets/javascripts/discourse/app/components/modal/spreadsheet-editor.gjs b/app/assets/javascripts/discourse/app/components/modal/spreadsheet-editor.gjs index 3b1b1696843..6d483e765eb 100644 --- a/app/assets/javascripts/discourse/app/components/modal/spreadsheet-editor.gjs +++ b/app/assets/javascripts/discourse/app/components/modal/spreadsheet-editor.gjs @@ -2,7 +2,6 @@ import Component from "@glimmer/component"; import { tracked } from "@glimmer/tracking"; import { action } from "@ember/object"; import didInsert from "@ember/render-modifiers/modifiers/did-insert"; -import { schedule } from "@ember/runloop"; import { inject as service } from "@ember/service"; import ConditionalLoadingSpinner from "discourse/components/conditional-loading-spinner"; import DButton from "discourse/components/d-button"; @@ -11,7 +10,6 @@ import DModalCancel from "discourse/components/d-modal-cancel"; import TextField from "discourse/components/text-field"; import { ajax } from "discourse/lib/ajax"; import { popupAjaxError } from "discourse/lib/ajax-error"; -import loadScript from "discourse/lib/load-script"; import { arrayToTable, findTableRegex, @@ -24,11 +22,16 @@ import DTooltip from "float-kit/components/d-tooltip"; export default class SpreadsheetEditor extends Component { @service dialog; @tracked showEditReason = false; - @tracked loading = null; + @tracked loading = true; spreadsheet = null; defaultColWidth = 150; isEditingTable = !!this.args.model.tableTokens; + constructor() { + super(...arguments); + this.loadJspreadsheet(); + } + get modalAttributes() { if (this.isEditingTable) { return { @@ -53,15 +56,11 @@ export default class SpreadsheetEditor extends Component { createSpreadsheet(spreadsheet) { this.spreadsheet = spreadsheet; - schedule("afterRender", () => { - this.loadLibraries().then(() => { - if (this.isEditingTable) { - this.buildPopulatedTable(this.args.model.tableTokens); - } else { - this.buildNewTable(); - } - }); - }); + if (this.isEditingTable) { + this.buildPopulatedTable(this.args.model.tableTokens); + } else { + this.buildNewTable(); + } } @action @@ -116,13 +115,15 @@ export default class SpreadsheetEditor extends Component { } } - loadLibraries() { - this.loading = true; - return loadScript("/javascripts/jsuites/jsuites.js") - .then(() => { - return loadScript("/javascripts/jspreadsheet/jspreadsheet.js"); - }) - .finally(() => (this.loading = false)); + async loadJspreadsheet() { + const [jspreadsheetModule] = await Promise.all([ + import("jspreadsheet-ce"), + import("jspreadsheet-ce/dist/jspreadsheet.css"), + import("jsuites/dist/jsuites.css"), + ]); + + this.jspreadsheet = jspreadsheetModule.default; + this.loading = false; } buildNewTable() { @@ -201,7 +202,7 @@ export default class SpreadsheetEditor extends Component { : `post-table-export`; // eslint-disable-next-line no-undef - this.spreadsheet = jspreadsheet(this.spreadsheet, { + this.spreadsheet = this.jspreadsheet(this.spreadsheet, { data, columns, defaultColAlign: "left", diff --git a/app/assets/javascripts/discourse/package.json b/app/assets/javascripts/discourse/package.json index 74487470ceb..86a21fd7505 100644 --- a/app/assets/javascripts/discourse/package.json +++ b/app/assets/javascripts/discourse/package.json @@ -24,6 +24,7 @@ "ember-source": "~3.28.12", "handlebars": "^4.7.8", "highlight.js": "^11.9.0", + "jspreadsheet-ce": "^4.13.4", "pretty-text": "1.0.0" }, "devDependencies": { diff --git a/app/assets/javascripts/yarn-ember3.lock b/app/assets/javascripts/yarn-ember3.lock index a2794726bab..834a0ff79e4 100644 --- a/app/assets/javascripts/yarn-ember3.lock +++ b/app/assets/javascripts/yarn-ember3.lock @@ -1560,6 +1560,11 @@ "@jridgewell/resolve-uri" "3.1.0" "@jridgewell/sourcemap-codec" "1.4.14" +"@jspreadsheet/formula@^2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@jspreadsheet/formula/-/formula-2.0.2.tgz#e0529f275c086fd7d34179b50450a98aeb56fd58" + integrity sha512-PDQYf9REQA53I7tVYkvkeyQxrd5jcjUeHgItYnRpjN2QiIQwawSqBDtGGEVQTSboTG+JwgGCuhvOpj7FxeKwew== + "@nodelib/fs.scandir@2.1.5": version "2.1.5" resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" @@ -7655,6 +7660,19 @@ jsonify@^0.0.1: resolved "https://registry.yarnpkg.com/jsonify/-/jsonify-0.0.1.tgz#2aa3111dae3d34a0f151c63f3a45d995d9420978" integrity sha512-2/Ki0GcmuqSrgFyelQq9M05y7PS0mEwuIzrf3f1fPqkVDVRvZrPZtVSMHxdgo8Aq0sxAOb/cr2aqqA3LeWHVPg== +jspreadsheet-ce@^4.13.4: + version "4.13.4" + resolved "https://registry.yarnpkg.com/jspreadsheet-ce/-/jspreadsheet-ce-4.13.4.tgz#78b11090a3dcc206b53d7dd72c030b6a5abdf8b4" + integrity sha512-Rv1xbR5AKme7Nd+vCRsHS05+3h0CtcDYcGseXPOEOWV9Mq7k3z57comq+kjLXJZyEf3CR9kCzIPQsd6tN7Yn6w== + dependencies: + "@jspreadsheet/formula" "^2.0.2" + jsuites "^5.0.25" + +jsuites@^5.0.25: + version "5.0.30" + resolved "https://registry.yarnpkg.com/jsuites/-/jsuites-5.0.30.tgz#85ff6e0847f9dbc574232cd34728eac8f53d719d" + integrity sha512-QFMgWH5pIxBxiXX8JB97gIUwsNazMn791fXCkX1HRXZdX9yKOC18InS0qd8ftkt2zTfdAgOG+t2cvteV4EUsEg== + just-extend@^4.0.2: version "4.2.1" resolved "https://registry.yarnpkg.com/just-extend/-/just-extend-4.2.1.tgz#ef5e589afb61e5d66b24eca749409a8939a8c744" diff --git a/app/assets/javascripts/yarn-ember5.lock b/app/assets/javascripts/yarn-ember5.lock index 2eff0d38c61..79b660607f7 100644 --- a/app/assets/javascripts/yarn-ember5.lock +++ b/app/assets/javascripts/yarn-ember5.lock @@ -1712,6 +1712,11 @@ "@jridgewell/resolve-uri" "3.1.0" "@jridgewell/sourcemap-codec" "1.4.14" +"@jspreadsheet/formula@^2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@jspreadsheet/formula/-/formula-2.0.2.tgz#e0529f275c086fd7d34179b50450a98aeb56fd58" + integrity sha512-PDQYf9REQA53I7tVYkvkeyQxrd5jcjUeHgItYnRpjN2QiIQwawSqBDtGGEVQTSboTG+JwgGCuhvOpj7FxeKwew== + "@nodelib/fs.scandir@2.1.5": version "2.1.5" resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" @@ -7861,6 +7866,19 @@ jsonify@^0.0.1: resolved "https://registry.yarnpkg.com/jsonify/-/jsonify-0.0.1.tgz#2aa3111dae3d34a0f151c63f3a45d995d9420978" integrity sha512-2/Ki0GcmuqSrgFyelQq9M05y7PS0mEwuIzrf3f1fPqkVDVRvZrPZtVSMHxdgo8Aq0sxAOb/cr2aqqA3LeWHVPg== +jspreadsheet-ce@^4.13.4: + version "4.13.4" + resolved "https://registry.yarnpkg.com/jspreadsheet-ce/-/jspreadsheet-ce-4.13.4.tgz#78b11090a3dcc206b53d7dd72c030b6a5abdf8b4" + integrity sha512-Rv1xbR5AKme7Nd+vCRsHS05+3h0CtcDYcGseXPOEOWV9Mq7k3z57comq+kjLXJZyEf3CR9kCzIPQsd6tN7Yn6w== + dependencies: + "@jspreadsheet/formula" "^2.0.2" + jsuites "^5.0.25" + +jsuites@^5.0.25: + version "5.0.30" + resolved "https://registry.yarnpkg.com/jsuites/-/jsuites-5.0.30.tgz#85ff6e0847f9dbc574232cd34728eac8f53d719d" + integrity sha512-QFMgWH5pIxBxiXX8JB97gIUwsNazMn791fXCkX1HRXZdX9yKOC18InS0qd8ftkt2zTfdAgOG+t2cvteV4EUsEg== + just-extend@^4.0.2: version "4.2.1" resolved "https://registry.yarnpkg.com/just-extend/-/just-extend-4.2.1.tgz#ef5e589afb61e5d66b24eca749409a8939a8c744" diff --git a/app/assets/stylesheets/common/table-builder/_index.scss b/app/assets/stylesheets/common/table-builder/_index.scss index 34264181789..3e993c55e22 100644 --- a/app/assets/stylesheets/common/table-builder/_index.scss +++ b/app/assets/stylesheets/common/table-builder/_index.scss @@ -1,5 +1,3 @@ -@import "vendor/jspreadsheet"; -@import "vendor/jsuites"; @import "jspreadsheet-theme"; @import "table-edit-decorator"; @import "insert-table-modal"; diff --git a/app/assets/stylesheets/common/table-builder/vendor/jspreadsheet.scss b/app/assets/stylesheets/common/table-builder/vendor/jspreadsheet.scss deleted file mode 100644 index 92572f0831f..00000000000 --- a/app/assets/stylesheets/common/table-builder/vendor/jspreadsheet.scss +++ /dev/null @@ -1,699 +0,0 @@ -:root { - --jexcel-border-color: #000; -} - -.jexcel_container { - display: inline-block; - padding-right: 2px; - box-sizing: border-box; - overscroll-behavior: contain; - outline: none; -} - -.jexcel_container.fullscreen { - position: fixed; - top: 0px; - left: 0px; - width: 100%; - height: 100%; - z-index: 21; -} - -.jexcel_container.fullscreen .jexcel_content { - overflow: auto; - width: 100%; - height: 100%; - background-color: #ffffff; -} - -.jexcel_container.with-toolbar .jexcel > thead > tr > td { - top: 0; -} - -.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: 8px; - height: 8px; -} - -.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; -} - -.jexcel_container.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: relative; - 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_overflow > tbody > tr > td { - overflow: hidden; -} - -.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; - padding-right: 2px; - 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); -} -.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: 20; - 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: 0.875rem; - letter-spacing: 0.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: 0.875rem; - letter-spacing: 0.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: 8px; - 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 4px 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: 10px; - padding-left: 20px; - padding-right: 20px; - margin-right: 5px; - margin-bottom: 5px; - 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 > tfoot > 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, -.jexcel tfoot .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; -} - -.red { - color: red; -} - -.jexcel > tbody > tr > td.readonly > input[type="checkbox"], -.jexcel > tbody > tr > td.readonly > input[type="radio"] { - pointer-events: none; - opacity: 0.5; -} diff --git a/app/assets/stylesheets/common/table-builder/vendor/jsuites.scss b/app/assets/stylesheets/common/table-builder/vendor/jsuites.scss deleted file mode 100644 index dcbd6c05b85..00000000000 --- a/app/assets/stylesheets/common/table-builder/vendor/jsuites.scss +++ /dev/null @@ -1,3038 +0,0 @@ -.jexcel_container { - :root { - --button-color: #298ba8; - --active-color: #007aff; - --safe-area-top: env(safe-area-inset-top); - --safe-area-bottom: env(safe-area-inset-bottom); - } - - [data-visible="false"] { - display: none; - } - - div[data-before]:before { - content: attr(data-before); - } - - .unselectable { - -webkit-touch-callout: none; - -webkit-user-select: none; - -khtml-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - } - - .jreadonly { - pointer-events: none; - } - - .jdragging { - opacity: 0.2; - filter: alpha(opacity=20); - } - - .jupload.input { - position: relative; - box-sizing: border-box; - background-size: initial; - height: 33px; - min-height: initial; - padding: 6px; - padding-right: 30px; - } - - .jupload.input:before { - content: "save"; - font-size: 18px; - font-family: "Material Icons"; - color: #000; - position: absolute; - right: 5px; - } - - .jupload:empty:before { - z-index: 0; - } - - .jupload img { - width: 100%; - } - - .jupload.input img { - width: initial; - max-width: 100%; - height: 100%; - } - - .jupload[data-multiple] { - padding: 10px; - } - - .jupload[data-multiple] img { - height: 70px; - width: 100px; - object-fit: cover; - margin-right: 5px; - margin-bottom: 5px; - } - - .jupload { - position: relative; - border: 1px dotted #eee; - cursor: pointer; - box-sizing: border-box; - width: 100%; - max-width: 100%; - max-height: 100%; - min-height: 180px; - } - - .jupload:not(.input):before { - content: "\e2c3"; - font-family: "Material Icons"; - font-size: 90px; - color: #eee; - width: 100%; - height: 100%; - display: flex; - align-items: center; - justify-content: center; - position: absolute; - z-index: -1; - } - - .jupload-item { - padding-right: 22px; - border-radius: 1px; - display: inline-block; - position: relative; - } - - .jphoto { - position: relative; - border: 1px dotted #eee; - cursor: pointer; - box-sizing: border-box; - width: 100%; - display: flex; - align-items: center; - justify-content: center; - } - - .jphoto:empty:before { - content: "\e2c3"; - font-family: "Material Icons"; - font-size: 90px; - color: #eee; - width: 100%; - height: 100%; - text-align: center; - } - - .jremove { - opacity: 0.2; - filter: alpha(opacity=20); - } - - .round img { - border-radius: 1000px; - } - - .jtooltip { - position: fixed; - top: 10px; - left: 10px; - z-index: 5; - - font-family: initial; - font-size: 12px; - color: #000; - background-color: #fff; - border: 1px solid black; - padding: 8px; - margin: 10px; - - display: block; - animation: fadeIn 0.5s; - pointer-events: none; - } - - .jtooltip:empty { - display: none; - } - - @keyframes fadeIn { - 0% { - opacity: 0; - } - 100% { - opacity: 1; - } - } - /** Loading */ - .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 0.8s linear infinite; - animation: spin 0.8s linear infinite; - } - - .jloading.spin { - background-color: transparent; - } - - .jloading.spin::after { - margin: 0 auto; - margin-top: 80px; - border-color: #aaa; - border-top-color: transparent; - } - - /** Animations **/ - .fade-in { - animation: fade-in 2s forwards; - } - - .fade-out { - animation: fade-out 1s forwards; - } - - .slide-left-in { - position: relative; - animation: slide-left-in 0.4s forwards; - } - - .slide-left-out { - position: relative; - animation: slide-left-out 0.4s forwards; - } - - .slide-right-in { - position: relative; - animation: slide-right-in 0.4s forwards; - } - - .slide-right-out { - position: relative; - animation: slide-right-out 0.4s forwards; - } - - .slide-top-in { - position: relative; - animation: slide-top-in 0.4s forwards; - } - - .slide-top-out { - position: relative; - animation: slide-top-out 0.2s forwards; - } - - .slide-bottom-in { - position: relative; - animation: slide-bottom-in 0.4s forwards; - } - - .slide-bottom-out { - position: relative; - animation: slide-bottom-out 0.1s forwards; - } - - .slide-left-in > div { - -webkit-transform: translateZ(0px); - -webkit-transform: translate3d(0, 0, 0); - } - - .slide-left-out > div { - -webkit-transform: translateZ(0px); - -webkit-transform: translate3d(0, 0, 0); - } - - .slide-right-in > div { - -webkit-transform: translateZ(0px); - -webkit-transform: translate3d(0, 0, 0); - } - - .slide-right-out > div { - -webkit-transform: translateZ(0px); - -webkit-transform: translate3d(0, 0, 0); - } - - .spin { - animation: spin 2s infinite linear; - } - - /** 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% { - left: -100%; - } - 100% { - left: 0%; - } - } - - @-webkit-keyframes slide-left-in { - 0% { - left: -100%; - } - 100% { - left: 0%; - } - } - - @keyframes slide-left-out { - 0% { - left: 0%; - } - 100% { - left: -100%; - } - } - - @-webkit-keyframes slide-left-out { - 0% { - left: 0%; - } - 100% { - left: -100%; - } - } - - /** Keyframes Right to Left **/ - @keyframes slide-right-in { - 0% { - left: 100%; - } - 100% { - left: 0%; - } - } - - @-webkit-keyframes slide-right-in { - 0% { - left: 100%; - } - 100% { - left: 0%; - } - } - - @keyframes slide-right-out { - 0% { - left: 0%; - } - 100% { - left: 100%; - } - } - - @-webkit-keyframes slide-right-out { - 0% { - left: 0%; - } - 100% { - left: 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%); - } - } - - @-webkit-keyframes spin { - from { - -webkit-transform: rotate(0deg); - } - to { - -webkit-transform: rotate(359deg); - } - } - - @keyframes spin { - from { - transform: rotate(0deg); - } - to { - transform: rotate(359deg); - } - } - .jcalendar { - position: absolute; - 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.jcalendar-focus { - display: block; - } - - .jcalendar .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 .jcalendar-container { - position: relative; - box-sizing: border-box; - } - - .jcalendar .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-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=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27%3E%3Cpath d=%27M15.41 16.59L10.83 12l4.58-4.59L14 6l-6 6 6 6 1.41-1.41z%27 fill=%27%23000%27 /%3E%3Cpath fill=%27none%27 d=%27M0 0h24v24H0V0z%27/%3E%3C/svg%3E"); - background-position: center; - background-repeat: no-repeat; - } - - .jcalendar-next { - cursor: pointer; - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27%3E%3Cpath d=%27M8.59 16.59L13.17 12 8.59 7.41 10 6l6 6-6 6-1.41-1.41z%27 fill=%27%23000%27 /%3E%3Cpath fill=%27none%27 d=%27M0 0h24v24H0V0z%27/%3E%3C/svg%3E"); - background-position: center; - background-repeat: no-repeat; - } - - .jcalendar-weekday { - font-weight: 600; - background-color: #fcfcfc; - padding: 14px; - } - - .jcalendar-table { - padding: 10px; - } - - .jcalendar-table > table { - width: 100%; - background-color: #fff; - } - - .jcalendar-table > table > thead { - cursor: pointer; - } - - .jcalendar-table thead td { - padding: 10px; - height: 40px; - } - - .jcalendar-table > table > tbody > tr { - height: 34px; - } - - .jcalendar-table > table > tbody td { - box-sizing: border-box; - cursor: pointer; - padding: 9px; - font-size: 0.9em; - } - - .jcalendar-table 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=%27http://www.w3.org/2000/svg%27 width=%2716%27 height=%2716%27 viewBox=%270 0 24 24%27 fill=%27gray%27%3E%3Cpath d=%27M20 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%27/%3E%3Cpath fill=%27none%27 d=%27M0 0h24v24H0z%27/%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-container select { - width: 55px; - display: inline-block; - border: 0px; - padding: 4px; - text-align: center; - font-size: 1.1em; - user-select: none; - margin-right: 10px; - } - - .jcalendar-container 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; - } - - .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; - } - - .jcalendar-time { - display: flex; - } - - .jcalendar_warning { - color: red; - } - - .jcalendar-hide-controls .jcalendar-controls { - display: none; - } - - .jcolor { - display: none; - outline: none; - position: absolute; - } - - .jcolor-input { - padding-right: 24px !important; - background: url("data:image/svg+xml,%0A%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 viewBox=%270 0 24 24%27 fill=%27black%27 width=%2718px%27 height=%2718px%27%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3Cpath d=%27M12 3c-4.97 0-9 4.03-9 9s4.03 9 9 9c.83 0 1.5-.67 1.5-1.5 0-.39-.15-.74-.39-1.01-.23-.26-.38-.61-.38-.99 0-.83.67-1.5 1.5-1.5H16c2.76 0 5-2.24 5-5 0-4.42-4.03-8-9-8zm-5.5 9c-.83 0-1.5-.67-1.5-1.5S5.67 9 6.5 9 8 9.67 8 10.5 7.33 12 6.5 12zm3-4C8.67 8 8 7.33 8 6.5S8.67 5 9.5 5s1.5.67 1.5 1.5S10.33 8 9.5 8zm5 0c-.83 0-1.5-.67-1.5-1.5S13.67 5 14.5 5s1.5.67 1.5 1.5S15.33 8 14.5 8zm3 4c-.83 0-1.5-.67-1.5-1.5S16.67 9 17.5 9s1.5.67 1.5 1.5-.67 1.5-1.5 1.5z%27/%3E%3C/svg%3E") - top 50% right 4px no-repeat, - content-box; - box-sizing: border-box; - } - - .jcolor-content { - position: absolute; - z-index: 9000; - user-select: none; - -webkit-font-smoothing: antialiased; - font-size: 0.875rem; - letter-spacing: 0.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; - min-width: 260px; - } - - .jmodal .jcolor-content { - position: fixed; - } - - .jcolor-controls { - display: flex; - padding: 10px; - border-bottom: 1px solid #eee; - margin-bottom: 5px; - } - - .jcolor-controls div { - flex: 1; - font-size: 1em; - color: var(--active-color); - text-transform: uppercase; - font-weight: bold; - box-sizing: border-box; - } - - .jcolor-content table { - border-collapse: collapse; - box-sizing: border-box; - } - - .jcolor-focus { - display: block; - } - - .jcolor table { - width: 100%; - height: 100%; - min-height: 160px; - } - - .jcolor td { - padding: 7px; - } - - .jcolor-selected { - background-repeat: no-repeat; - background-size: 16px; - background-position: center; - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3Cpath d=%27M9 16.17L4.83 12l-1.42 1.41L9 19 21 7l-1.41-1.41z%27 fill=%27white%27/%3E%3C/svg%3E"); - } - - .jcolor-fullscreen { - position: fixed; - left: 0px; - bottom: 0px; - width: 100%; - max-height: 290px; - border-radius: 0px; - box-sizing: border-box; - } - - .jcolor-fullscreen .jcolor-controls { - padding: 15px; - -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); - } - - .jcolor-reset { - text-align: left; - } - - .jcolor-close { - text-align: right; - } - - .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 */ - } - - .jcolor-content .jtabs-content { - padding: 7px; - } - - .jcolor-grid tr:first-child > td:first-child { - border-top-left-radius: 3px; - } - - .jcolor-grid tr:first-child > td:last-child { - border-top-right-radius: 3px; - } - - .jcolor-grid tr:last-child > td:first-child { - border-bottom-left-radius: 3px; - } - - .jcolor-grid tr:last-child > td:last-child { - border-bottom-right-radius: 3px; - } - - .jcolor-hsl { - box-sizing: border-box; - } - - .jcolor-hsl > div { - height: 100%; - position: relative; - } - - .jcolor-hsl canvas { - display: block; - border-radius: 4px; - -webkit-user-drag: none; - } - - .jcolor-point { - height: 5px; - width: 5px; - background-color: #000; - position: absolute; - top: 50%; - left: 50%; - transform: translate(-50%, -50%); - border-radius: 50%; - } - - .jcolor-sliders { - padding: 10px 20px 10px 10px; - } - - .jcolor-sliders input { - -webkit-appearance: none; - - height: 12px; - width: 80%; - - background: #d3d3d3; - opacity: 1; - - border-radius: 30px; - outline: none; - } - - .jcolor-sliders-input-subcontainer { - display: flex; - justify-content: space-between; - align-items: center; - } - - .jcolor-sliders-input-container { - margin-top: 4px; - line-height: 0.8em; - text-align: left; - } - - .jcolor-sliders-input-container > label { - font-size: 10px; - text-transform: uppercase; - color: #bbbbbd; - } - - .jcolor-sliders-input-subcontainer > input { - border: 0px; - padding: 1px; - } - - .jcolor-sliders-input-container input::-webkit-slider-thumb { - -webkit-appearance: none; - height: 12px; - width: 12px; - border-radius: 50%; - background: #000; - border: 2px solid #fff; - cursor: pointer; - } - - .jcolor-sliders-input-container input::-moz-range-thumb { - -webkit-appearance: none; - height: 12px; - width: 12px; - border-radius: 50%; - background: #000; - border: 2px solid #fff; - cursor: pointer; - } - - .jcolor-sliders-final-color { - padding: 6px; - user-select: all; - margin-top: 10px; - text-align: center; - } - - .jcolor-sliders-final-color > div:nth-child(2) { - width: 71px; - text-transform: uppercase; - } - - .jcolor .jtabs .jtabs-headers-container .jtabs-controls { - display: none !important; - } - - .jcolor .jtabs .jtabs-headers-container { - display: flex !important; - justify-content: center; - padding: 4px; - } - - .jcolor .jtabs-headers > div:not(.jtabs-border) { - padding: 2px !important; - padding-left: 15px !important; - padding-right: 15px !important; - font-size: 0.8em; - } - .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: flex; - padding: 8px 8px 8px 32px; - width: 250px; - position: relative; - cursor: default; - font-size: 11px; - font-family: sans-serif; - text-align: left; - -webkit-box-align: center; - align-items: center; - } - - .jcontextmenu > div::before { - content: attr(data-icon); - font-family: "Material Icons" !important; - font-size: 15px; - position: absolute; - left: 9px; - line-height: 24px; - } - - .jcontextmenu.symbols > div::before { - font-family: "Material Symbols Outlined" !important; - } - - .jcontextmenu > div.header { - display: none; - } - - .jcontextmenu > div a { - color: #555; - text-decoration: none; - flex: 1; - cursor: pointer; - } - - .jcontextmenu > div span { - margin-right: 10px; - } - - .jcontextmenu .jcontextmenu-disabled a { - color: #ccc; - } - - .jcontextmenu .jcontextmenu-disabled::before { - color: #ccc; - } - - .jcontextmenu > div:hover { - background: #ebebeb; - } - - .jcontextmenu hr { - border: 1px solid #e9e9e9; - border-bottom: 0; - margin-top: 5px; - margin-bottom: 5px; - } - - .jcontextmenu > hr:hover { - background: transparent; - } - - .jcontextmenu .jcontextmenu { - top: 4px; - left: 99%; - opacity: 0; - position: absolute; - } - - .jcontextmenu > div:hover > .jcontextmenu { - display: block; - opacity: 1; - -webkit-transform: translate(0, 0) scale(1); - transform: translate(0, 0) scale(1); - pointer-events: auto; - } - - @media only screen and (max-width: 420px) { - .jcontextmenu { - top: initial !important; - left: 0px !important; - bottom: 0px !important; - width: 100vw; - height: 260px; - overflow: scroll; - animation: slide-bottom-in 0.4s forwards; - padding-top: 0px; - } - .jcontextmenu div { - width: 100%; - text-align: center; - border-bottom: 1px solid #ccc; - padding: 15px; - } - .jcontextmenu > div.header { - background-color: lightgray; - padding: 5px; - top: 0px; - position: sticky; - z-index: 2; - } - .jcontextmenu > div.header > a.title { - text-align: left; - } - - .jcontextmenu > div.header > a.close { - text-align: right; - } - .jcontextmenu a { - font-size: 1.4em; - text-transform: uppercase; - } - .jcontextmenu span { - display: none; - } - .jcontextmenu span { - display: none; - } - .jcontextmenu hr { - display: none; - } - } - - .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-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[disabled] { - opacity: 0.5; - pointer-events: none; - } - - .jdropdown-focus { - position: relative; - } - - .jdropdown-focus .jdropdown-container { - transform: translate3d(0, 0, 0); - } - - .jdropdown-default.jdropdown-focus .jdropdown-header { - outline: auto 5px -webkit-focus-ring-color; - } - - .jdropdown-default.jdropdown-focus .jdropdown-header.jdropdown-add { - background-image: url("data:image/svg+xml,%0A%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 viewBox=%270 0 24 24%27 fill=%27gray%27 width=%2724px%27 height=%2724px%27%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3Cpath d=%27M19 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%27/%3E%3C/svg%3E"); - } - - .jdropdown-container-header { - padding: 0px; - margin: 0px; - position: relative; - box-sizing: border-box; - } - - .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=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27%3E%3Cpath fill=%27none%27 d=%27M0 0h24v24H0V0z%27/%3E%3Cpath d=%27M7 10l5 5 5-5H7z%27 fill=%27gray%27/%3E%3C/svg%3E"); - text-overflow: ellipsis; - cursor: pointer; - box-sizing: border-box; - -webkit-appearance: none; - -moz-appearance: none; - padding-right: 30px !important; - } - - .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: 12px; - 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-description { - text-overflow: ellipsis; - overflow: hidden; - line-height: 1.5em; - } - - .jdropdown-image { - margin-right: 10px; - width: 32px; - height: 32px; - border-radius: 20px; - } - - .jdropdown-image-small { - width: 24px; - height: 24px; - } - - .jdropdown-icon { - margin-right: 10px; - font-size: 30px; - margin-left: -5px; - } - - .jdropdown-icon-small { - font-size: 24px; - margin-left: 0px; - } - - .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: #eee; - } - - .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; - text-align: left; - } - - .jdropdown-default .jdropdown-reset_ { - content: "x"; - position: absolute; - top: 0; - right: 0; - margin: 5px; - margin-right: 10px; - font-size: 12px; - width: 12px; - cursor: pointer; - text-shadow: 0px 0px 5px #fff; - display: none; - line-height: 1.8em; - } - - .jdropdown-default.jdropdown-focus .jdropdown-reset_ { - display: block; - } - - /** Default render for mobile **/ - - .jdropdown-picker.jdropdown-focus .jdropdown-backdrop { - display: block; - } - - .jdropdown-picker .jdropdown-header { - outline: none; - } - - .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 !important; - left: 0px !important; - width: 100% !important; - height: 100% !important; - background-color: #fafafa; - padding: 0px; - z-index: 9001; - overflow-y: scroll; - will-change: scroll-position; - -webkit-overflow-scrolling: touch; - } - - .jdropdown-searchbar.jdropdown-focus .jdropdown-container-header { - position: fixed; - top: 0px; - left: 0px; - z-index: 9002; - padding: 6px; - background-color: #fff; - box-shadow: 0 1px 2px rgba(0, 0, 0, 0.1); - width: 100%; - } - - .jdropdown-searchbar.jdropdown-focus .jdropdown-header { - border: 0px !important; - background-position-x: 0% !important; - background-position-y: 40% !important; - background-repeat: no-repeat; - background-image: url(); - padding-left: 30px !important; - padding-right: 60px !important; - } - - .jdropdown-searchbar.jdropdown-focus .jdropdown-close { - display: block; - } - - .jdropdown-searchbar .jdropdown-header { - outline: none; - } - - .jdropdown-searchbar .jdropdown-container { - margin-top: 40px; - 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-width: 800px) { - .jdropdown-list { - width: 100% !important; - border: 0px; - padding: 0px; - } - - .jdropdown-list .jdropdown-container { - min-width: 100%; - } - - .jdropdown-searchbar.jdropdown-focus .jdropdown-description { - text-transform: uppercase; - } - } - - .app .jdropdown-item { - text-transform: uppercase; - } - - .jdropdown-create-container { - margin: 10px; - border: 1px solid #ccc; - border-radius: 2px; - padding: 6px; - } - - .jdropdown-color { - background-color: #fff; - border: 1px solid transparent; - border-radius: 12px; - width: 12px; - height: 12px; - margin-right: 6px; - } - - .jdropdown-item[data-disabled] { - opacity: 0.5; - pointer-events: none; - } - .jeditor-container { - border: 1px solid #ccc; - box-sizing: border-box; - } - - .jeditor-container.with-margin { - background-color: #f2f2f2; - max-width: 1200px; - } - - .jeditor-dragging { - border: 1px dashed #000; - } - - .jeditor { - outline: none; - word-break: break-word; - } - - .jeditor-container.with-margin .jeditor { - background-color: #fff; - margin: 80px; - min-height: 800px; - padding: 80px; - max-width: 800px; - } - - .jeditor[data-placeholder]:empty:before { - content: attr(data-placeholder); - color: lightgray; - } - - /** Snippet **/ - - .jsnippet { - margin-top: 15px; - cursor: pointer; - border: 1px solid #ccc; - position: relative; - } - - .jsnippet:focus { - outline: none; - } - - .jsnippet img { - width: 100%; - } - - .jsnippet .jsnippet-title { - padding: 15px; - font-size: 1.4em; - } - - .jsnippet .jsnippet-description { - padding-left: 15px; - padding-right: 15px; - font-size: 1em; - } - - .jsnippet .jsnippet-host { - padding: 15px; - text-transform: uppercase; - font-size: 0.8em; - color: #777; - text-align: right; - } - - .jsnippet .jsnippet-url { - display: none; - } - - .jeditor .jsnippet:after { - content: "close"; - font-family: "Material icons"; - font-size: 24px; - width: 24px; - height: 24px; - line-height: 24px; - cursor: pointer; - text-shadow: 0px 0px 2px #fff; - position: absolute; - top: 12px; - right: 12px; - } - - .jsnippet * { - -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.resizing { - -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:focus { - border: 2px solid #0096fd; - outline: #0096fd; - } - - .jeditor .pdf { - background-image: url("data:image/svg+xml,%3Csvg version=%271.1%27 id=%27Layer_1%27 xmlns=%27http://www.w3.org/2000/svg%27 xmlns:xlink=%27http://www.w3.org/1999/xlink%27 x=%270px%27 y=%270px%27 viewBox=%270 0 512 512%27 style=%27enable-background:new 0 0 512 512;%27 xml:space=%27preserve%27%3E%3Cpath style=%27fill:%23C30B15;%27 d=%27M511.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%27/%3E%3Cpath style=%27fill:%2385080E;%27 d=%27M511.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%27/%3E%3Cpolygon style=%27fill:%23FFFFFF;%27 points=%27278.328,333.913 255.711,77.913 119.096,77.913 119.096,311.652 %27/%3E%3Cpolygon style=%27fill:%23E8E6E6;%27 points=%27392.904,311.652 392.904,155.826 337.252,133.565 314.991,77.913 255.711,77.913 256.067,333.913 %27/%3E%3Cpolygon style=%27fill:%23FFFFFF;%27 points=%27314.991,155.826 314.991,77.913 392.904,155.826 %27/%3E%3Crect x=%27119.096%27 y=%27311.652%27 style=%27fill:%23FC0F1A;%27 width=%27273.809%27 height=%27122.435%27/%3E%3Cg%3E%3Cpath style=%27fill:%23FFFFFF;%27 d=%27M204.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%27/%3E%3Cpath style=%27fill:%23FFFFFF;%27 d=%27M257.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%27/%3E%3Cpath style=%27fill:%23FFFFFF;%27 d=%27M308.563,356.982v12.26h23.763v10.596h-23.763v19.525h-13.471v-52.975h39.277v10.595h-25.806 V356.982z%27/%3E%3C/g%3E%3C/svg%3E%0A"); - background-repeat: no-repeat; - background-size: cover; - width: 60px; - height: 60px; - } - - .jeditor-toolbar { - width: fit-content; - max-width: 100%; - box-sizing: border-box; - margin: 10px; - } - - .toolbar-on-top .jeditor-toolbar { - width: initial; - margin: 0px; - box-shadow: 1px 1px 2px rgba(0, 0, 0, 0.1); - display: block; - } - - .toolbar-on-top .jeditor { - padding: 15px; - } - - .toolbar-on-top .jtoolbar .material-icons { - font-size: 24px; - transform: initial; - margin: 4px; - } - - .toolbar-on-top .jtoolbar .jpicker-header { - font-size: 1em; - margin-top: 4px; - margin-bottom: 4px; - } - - .jeditor table { - border-collapse: collapse; - } - - .jeditor table td { - border: 1px solid #bbb; - height: 2em; - } - - .jeditor table td:focus { - border: 1px solid blue; - } - - .jeditor .line-break { - border-top: 1px dashed #ccc; - display: flex; - justify-content: center; - pointer-events: none; - } - - .jeditor .line-break:before { - content: "New page"; - background-color: #fff; - color: #ccc; - margin: -1em; - padding: 6px; - position: absolute; - } - .jfloating { - position: fixed; - bottom: 0px; - right: 0px; - margin-right: 5px; - - -webkit-box-shadow: 0 2px 10px rgba(0, 0, 0, 0.2); - -moz-box-shadow: 0 2px 10px rgba(0, 0, 0, 0.2); - border: 1px solid #ccc; - background-color: #fff; - box-sizing: border-box; - padding-top: 50px !important; - z-index: 9002; - border-radius: 8px; - } - - .jfloating.jfloating-big { - width: 510px !important; - height: 472px !important; - } - - .jfloating.jfloating-small { - width: 300px !important; - height: 320px !important; - } - - .jfloating.jfloating-large { - width: 600px !important; - height: 600px !important; - } - - .jfloating:before { - position: absolute; - top: 0; - left: 0; - width: 100%; - content: attr(title); - padding: 15px; - box-sizing: border-box; - font-size: 1.2em; - box-shadow: 1px 1px 3px rgba(0, 0, 0, 0.2); - background-color: #fff; - border-radius: 8px 8px 0px 0px; - background-color: #404040; - font-size: 0.93rem; - font-weight: 600; - color: white; - letter-spacing: 0.5px; - } - - .jfloating:after { - content: ""; - background-image: url("data:image/svg+xml,%0A%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27%3E%3Cpath fill=%27%23FFF%27 d=%27M19 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%27/%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%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; - } - - .jfloating_content { - padding: 20px; - overflow-y: auto; - max-height: 100%; - box-sizing: border-box; - height: -webkit-fill-available; - } - - .jfloating.jfloating-minimized { - height: 50px !important; - } - - .jfloating.jfloating-minimized .jfloating_content { - display: none; - } - - .jmodal { - position: fixed; - top: 50%; - left: 50%; - width: 60%; - height: 60%; - -webkit-box-shadow: 0 2px 12px rgba(0, 0, 0, 0.2); - -moz-box-shadow: 0 2px 12px rgba(0, 0, 0, 0.2); - border: 1px solid #ccc; - background-color: #fff; - transform: translate(-50%, -50%); - box-sizing: border-box; - z-index: 9002; - border-radius: 4px; - display: flex; - flex-direction: column; - } - - .jmodal_title { - padding: 20px; - height: 70px; - box-sizing: border-box; - font-size: 1.4em; - background-color: #fff; - border-radius: 8px 8px 0px 0px; - pointer-events: none; - display: flex; - -webkit-align-items: center; - -webkit-box-align: center; - align-items: center; - border-bottom: 1px solid #eee; - } - - .jmodal_title > div { - font-size: 1.4em; - } - - .jmodal_title[data-icon]:before { - content: attr(data-icon); - font-family: "Material Icons" !important; - width: 24px; - height: 24px; - font-size: 24px; - margin-right: 10px; - line-height: 24px; - } - - .jmodal_content { - padding: 20px; - overflow-y: auto; - height: 100%; - box-sizing: border-box; - scrollbar-width: thin; - scrollbar-color: #333 transparent; - } - - .jmodal_title:empty { - display: none; - } - - .jmodal_title:empty + .jmodal_content { - height: 100%; - } - - .jmodal_content::-webkit-scrollbar { - height: 12px; - } - - .jmodal_content::-webkit-scrollbar { - width: 12px; - } - - .jmodal_content::-webkit-scrollbar-track { - border: 1px solid #fff; - background: #eee; - } - - .jmodal_content::-webkit-scrollbar-thumb { - border: 1px solid #fff; - background: #888; - } - - .jmodal:after { - content: ""; - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27%3E%3Cpath d=%27M19 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%27/%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3C/svg%3E"); - position: absolute; - top: 0; - right: 0; - margin: 25px; - 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: 0px; - border-radius: 0px; - } - - .jmodal_backdrop { - position: fixed; - top: 0px; - left: 0px; - min-width: 100%; - min-height: 100%; - background-color: rgba(0, 0, 0, 0.2); - 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 */ - } - - .jmodal_content .jcalendar .jcalendar-content, - .jmodal_content .jdropdown-container { - position: fixed; - } - - .jnotification { - position: fixed; - z-index: 10000; - -webkit-box-sizing: border-box; - box-sizing: border-box; - padding: 10px; - bottom: 0px; - } - - .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; - - 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-close { - content: ""; - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2720%27 height=%2720%27 viewBox=%270 0 24 24%27 fill=%27white%27%3E%3Cpath d=%27M19 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%27/%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%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; - } - - .jnotification-error .jnotification-container { - background: rgb(182, 38, 6); - background: linear-gradient( - 0deg, - rgba(170, 41, 13, 1) 0%, - rgba(149, 11, 11, 1) 100% - ); - } - - @media (max-width: 800px) { - .jnotification { - top: calc(0px + var(--safe-area-top)); - width: 100%; - } - .jnotification-container { - background: rgba(255, 255, 255, 0.95); - border: 1px solid #eee; - color: #444; - margin: 0px; - width: initial; - } - .jnotification-error .jnotification-container { - background: rgba(255, 255, 255, 0.95); - color: #790909; - } - .jnotification-close { - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2720%27 height=%2720%27 viewBox=%270 0 24 24%27 fill=%27black%27%3E%3Cpath d=%27M19 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%27/%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3C/svg%3E"); - } - } - - .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; - } - .jpicker { - cursor: pointer; - white-space: nowrap; - display: inline-flex; - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - outline: none; - position: relative; - } - - .jpicker-header { - background-repeat: no-repeat; - background-position: top 50% right 5px; - background-image: url("data:image/svg+xml,%0A%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27%3E%3Cpath fill=%27none%27 d=%27M0 0h24v24H0V0z%27/%3E%3Cpath d=%27M7 10l5 5 5-5H7z%27 fill=%27gray%27/%3E%3C/svg%3E"); - text-overflow: ellipsis; - cursor: pointer; - box-sizing: border-box; - text-align: left; - outline: none; - - line-height: 24px; - padding: 2px; - padding-left: 12px; - padding-right: 35px; - outline: none; - border-radius: 4px; - } - - .jpicker-header:hover { - background-color: #f2f2f2; - } - - .jpicker-content { - position: absolute; - top: 0; - display: none; - 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); - border-radius: 4px; - background-color: #fff; - padding: 4px; - z-index: 50; - text-align: left; - max-height: 250px; - scrollbar-width: thin; - scrollbar-color: #333 transparent; - } - - .jpicker-content::-webkit-scrollbar { - width: 8px; - } - - .jpicker-content::-webkit-scrollbar-track { - background: #eee; - } - - .jpicker-content::-webkit-scrollbar-thumb { - background: #888; - } - - .jpicker-content > div { - padding: 6px; - padding-left: 15px; - padding-right: 15px; - } - - .jpicker-focus > .jpicker-content { - display: block; - } - - .jpicker-content > div:hover { - background-color: #efefef; - } - - .jpicker-content > div:empty { - opacity: 0; - } - - .jpicker-header > i, - .jpicker-header > div { - display: block; - } - - .jpicker-focus > .jpicker-content.jpicker-columns { - display: flex !important ; - justify-content: center; - flex-wrap: wrap; - } - - .jpicker-focus .jpicker-content.jpicker-grid { - display: inline-grid; - } - - .jprogressbar { - 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; - box-sizing: border-box; - cursor: pointer; - border: 1px solid #ccc; - position: relative; - } - - .jprogressbar::before { - content: attr(data-value); - position: absolute; - margin: 5px; - margin-left: 10px; - } - - .jprogressbar-header::placeholder { - color: #000; - } - - .jprogressbar::focus { - outline: auto 5px -webkit-focus-ring-color; - } - - .jprogressbar > div { - background-color: #eee; - background-color: red; - box-sizing: border-box; - height: 31px; - } - .jrating { - display: flex; - } - .jrating > div { - width: 24px; - height: 24px; - line-height: 24px; - background-image: url("data:image/svg+xml,%0A%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27%3E%3Cpath d=%27M22 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%27 fill=%27gray%27/%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3C/svg%3E"); - } - - .jrating .jrating-over { - background-image: url("data:image/svg+xml,%0A%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27 fill=%27black%27%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3Cpath d=%27M12 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%27/%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3C/svg%3E"); - opacity: 0.7; - } - - .jrating .jrating-selected { - background-image: url("data:image/svg+xml,%0A%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27 fill=%27red%27%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3Cpath d=%27M12 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%27/%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3C/svg%3E"); - } - - .jsearch { - position: relative; - display: none; - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - } - - .jsearch_container { - position: absolute; - 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; - z-index: 1; - - -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 0.218s; - transition: opacity 0.218s; - background: #fff; - border: 1px solid rgba(0, 0, 0, 0.2); - cursor: pointer; - margin: 0; - min-width: 300px; - outline: none; - width: auto; - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - } - - .jsearch_container:empty:after { - content: attr(data-placeholder); - } - - .jsearch_container > 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; - - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - } - - .jsearch_container > div:hover { - background-color: #e8eaed; - } - - .jsearch_container > div > img { - width: 32px; - height: 32px; - user-select: none; - border-radius: 16px; - margin-right: 2px; - } - - .jsearch_container > div > div { - overflow: hidden; - text-overflow: ellipsis; - margin-left: 2px; - max-width: 300px; - white-space: nowrap; - user-select: none; - } - - .jsearch_container .selected { - background-color: #e8eaed; - } - .jslider { - outline: none; - } - - .jslider-focus { - width: 100% !important; - height: 100% !important; - } - - .jslider-focus img { - display: none; - } - - .jslider img { - width: 100px; - } - - .jslider-left::before { - position: fixed; - left: 15px; - top: 50%; - content: "arrow_back_ios"; - color: #fff; - width: 30px; - height: 30px; - font-family: "Material Icons"; - font-size: 30px; - /* before it was 0px 0px 0px #000 */ - text-shadow: 0px 0px 6px rgb(56, 56, 56); - text-align: center; - cursor: pointer; - } - - .jslider-right::after { - position: fixed; - right: 15px; - top: 50%; - content: "arrow_forward_ios"; - color: #fff; - width: 30px; - height: 30px; - font-family: "Material Icons"; - font-size: 30px; - /* before it was 0px 0px 0px #000 */ - text-shadow: 0px 0px 6px rgb(56, 56, 56); - text-align: center; - cursor: pointer; - } - - .jslider-close { - width: 24px; - height: 24px; - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 width=%2724%27 height=%2724%27 viewBox=%270 0 24 24%27 fill=%27white%27%3E%3Cpath d=%27M19 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%27/%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3C/svg%3E"); - position: fixed; - top: 15px; - right: 15px; - cursor: pointer; - z-index: 3000; - - display: block !important; - } - - .jslider-counter { - height: 24px; - background-color: transparent; - position: fixed; - left: 50%; - transform: translateX(-50%); - bottom: 15px; - cursor: pointer; - z-index: 3000; - - display: flex; - display: -webkit-flex; - -webkit-justify-content: center; - -webkit-align-items: center; - -webkit-flex-direction: row; - justify-content: center; - align-items: center; - flex-direction: row; - } - - .jslider-caption { - position: fixed; - max-width: 90vw; - text-overflow: ellipsis; - white-space: nowrap; - overflow: hidden; - top: 15px; - left: 15px; - z-index: 3000; - color: #fff; - font-size: 1rem; - - display: block !important; - } - - .jslider-counter div { - width: 10px; - height: 10px; - background: #fff; - border-radius: 50%; - margin: 0px 5px; - - display: block !important; - } - - .jslider-counter .jslider-counter-focus { - background-color: cornflowerblue; - pointer-events: none; - } - - .jslider-focus { - position: fixed; - left: 0; - top: 0; - width: 100%; - min-height: 100%; - max-height: 100%; - z-index: 2000; - margin: 0px; - box-sizing: border-box; - - background-color: rgba(0, 0, 0, 0.8); - -webkit-transition-duration: 0.05s; - transition-duration: 0.05s; - display: flex; - -ms-flex-align: center; - -webkit-align-items: center; - -webkit-box-align: center; - - align-items: center; - } - - .jslider-focus img { - width: 50vw; - height: auto; - box-sizing: border-box; - margin: 0 auto; - vertical-align: middle; - display: none; - } - - .jslider-focus img.jslider-vertical { - width: auto; - /* before it was 50vh */ - height: 80vh; - } - - @media only screen and (max-width: 576px) { - .jslider-focus img.jslider-vertical { - width: 99vw !important; - height: auto !important; - } - - .jslider-focus img { - width: 100vw !important; - height: auto !important; - } - } - - .jslider-grid { - display: -ms-grid; - display: grid; - grid-gap: 1px; - position: relative; - } - - .jslider-grid[data-number="2"] { - -ms-grid-columns: 1fr 50%; - grid-template-columns: 1fr 50%; - } - - .jslider-grid[data-number="3"] { - -ms-grid-columns: 1fr 33%; - grid-template-columns: 1fr 33%; - } - - .jslider-grid[data-number="4"] { - -ms-grid-columns: 1fr 25%; - grid-template-columns: 1fr 25%; - } - - .jslider-grid img { - display: none; - width: 100%; - height: 100%; - object-fit: cover; - } - - .jslider-grid[data-total]:after { - content: attr(data-total) "+"; - font-size: 1.5em; - position: absolute; - color: #fff; - right: 15px; - bottom: 6px; - } - - .jslider-grid img:first-child { - -ms-grid-column: 1; - -ms-grid-row: 1; - grid-column: 1; - grid-row: 1; - display: block; - } - - .jslider-grid[data-number="2"] img:nth-child(2) { - -ms-grid-column: 2; - -ms-grid-row: 1; - grid-column: 2; - grid-row: 1; - display: block; - } - - .jslider-grid[data-number="3"] img:first-child { - -ms-grid-column: 1 / 2; - -ms-grid-row: 1 / 4; - grid-column: 1 / 2; - grid-row: 1 / 4; - } - - .jslider-grid[data-number="3"] img:nth-child(2) { - -ms-grid-column: 2; - -ms-grid-row: 1; - grid-column: 2; - grid-row: 1; - display: block; - } - - .jslider-grid[data-number="3"] img:nth-child(3) { - -ms-grid-column: 2; - -ms-grid-row: 2; - grid-column: 2; - grid-row: 2; - display: block; - } - - .jslider-grid[data-number="4"] img:first-child { - -ms-grid-column: 1 / 2; - -ms-grid-row: 1 / 4; - grid-column: 1 / 2; - grid-row: 1 / 4; - } - - .jslider-grid[data-number="4"] img:nth-child(2) { - -ms-grid-column: 2; - -ms-grid-row: 1; - grid-column: 2; - grid-row: 1; - display: block; - } - - .jslider-grid[data-number="4"] img:nth-child(3) { - -ms-grid-column: 2; - -ms-grid-row: 2; - grid-column: 2; - grid-row: 2; - display: block; - } - - .jslider-grid[data-number="4"] img:nth-child(4) { - -ms-grid-column: 2; - -ms-grid-row: 3; - grid-column: 2; - grid-row: 3; - display: block; - } - - .jtabs { - max-width: 100vw; - position: relative; - } - - .jtabs .jtabs-headers-container { - display: flex; - align-items: center; - } - - .jtabs .jtabs-headers { - display: flex; - align-items: center; - overflow: hidden; - position: relative; - } - - .jtabs .jtabs-headers > div:not(.jtabs-border) { - padding: 8px; - padding-left: 20px; - padding-right: 20px; - margin-left: 1px; - margin-right: 1px; - background-color: #f1f1f1; - cursor: pointer; - white-space: nowrap; - text-align: center; - } - - .jtabs .jtabs-headers > div.jtabs-selected { - background-color: #e8e8e8; - color: #000; - } - - .jtabs .jtabs-headers > div > div { - color: #555; - width: 100%; - overflow: hidden; - } - - .jtabs .jtabs-headers i { - display: block; - margin: auto; - } - - .jtabs .jtabs-content { - box-sizing: border-box; - } - - .jtabs .jtabs-content > div { - display: none; - box-sizing: border-box; - } - - .jtabs .jtabs-content > div.jtabs-selected { - display: block; - } - - .jtabs .jtabs-border { - position: absolute; - height: 2px; - background-color: #888; - transform-origin: left; - transition: all 0.2s cubic-bezier(0.4, 0, 0.2, 1); - transition-property: color, left, transform; - display: none; - } - - .jtabs-animation .jtabs-border { - display: initial; - } - - .jtabs .jtabs-controls { - margin: 3px; - margin-left: 10px; - display: flex; - min-width: 82px; - } - - .jtabs .jtabs-controls > div { - cursor: pointer; - background-position: center; - background-repeat: no-repeat; - width: 24px; - height: 24px; - line-height: 24px; - } - - .jtabs .jtabs-prev { - margin-left: 10px; - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 viewBox=%270 0 24 24%27 fill=%27gray%27 width=%2718px%27 height=%2718px%27%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3Cpath d=%27M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z%27/%3E%3C/svg%3E"); - } - - .jtabs .jtabs-prev.disabled { - margin-left: 10px; - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 viewBox=%270 0 24 24%27 fill=%27lightgray%27 width=%2718px%27 height=%2718px%27%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3Cpath d=%27M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z%27/%3E%3C/svg%3E"); - } - - .jtabs .jtabs-next { - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 viewBox=%270 0 24 24%27 fill=%27gray%27 width=%2718px%27 height=%2718px%27%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3Cpath d=%27M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z%27/%3E%3C/svg%3E"); - } - - .jtabs .jtabs-next.disabled { - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 viewBox=%270 0 24 24%27 fill=%27lightgray%27 width=%2718px%27 height=%2718px%27%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3Cpath d=%27M10 6L8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z%27/%3E%3C/svg%3E"); - } - - .jtabs .jtabs-add { - background-image: url("data:image/svg+xml,%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 height=%2724%27 viewBox=%270 0 24 24%27 width=%2724%27%3E%3Cpath d=%27M19 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%27 fill=%27%23bbbbbb%27/%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3C/svg%3E"); - } - - /** Modern skin **/ - - .jtabs.jtabs-modern .jtabs-headers > div:not(.jtabs-border) { - padding: 4px; - padding-left: 10px; - padding-right: 10px; - background-color: #fff; - } - - .jtabs.jtabs-modern .jtabs-headers > .jtabs-selected { - color: #000; - } - - .jtabs.jtabs-modern .jtabs-headers > .jtabs-selected .material-icons { - color: #000; - } - - .jtabs.jtabs-modern .jtabs-headers { - background: #eeeeef !important; - padding: 2px; - border-radius: 4px; - } - - .jtabs.jtabs-modern .jtabs-headers .jtabs-border { - border-color: #eeeeef !important; - } - - .jtabs.jtabs-modern .jtabs-border { - background-color: rgba(194, 197, 188, 0.884); - } - .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: 1px; - border: 1px solid #ccc; - position: relative; - } - - .jtags.jtags-empty:not(.jtags-focus)::before { - position: absolute; - margin: 3px; - color: #ccc; - content: attr(data-placeholder); - top: 0px; - margin-left: 6px; - } - - .jtags > div { - padding: 5px; - padding-left: 10px; - padding-right: 22px; - position: relative; - border-radius: 1px; - margin: 2px; - display: block; - outline: none; - } - - .jtags > div:empty:before { - content: " "; - white-space: pre; - } - - .jtags > div::after { - content: "x"; - position: absolute; - top: 6px; - 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; - } - - .jtoolbar-container { - border-radius: 2px; - 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); - display: inline-flex !important; - } - - .jtoolbar { - cursor: pointer; - white-space: nowrap; - display: flex; - padding: 4px; - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - font-size: 13px; - } - - .jtoolbar-disabled { - pointer-events: none; - opacity: 0.4; - } - - .jtoolbar-mobile { - display: flex; - position: fixed; - bottom: 0; - margin: 0; - left: 0; - width: 100%; - background: #f7f7f8; - z-index: 1; - box-sizing: border-box; - box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.1); - border-radius: 0px; - } - - .jtoolbar > div { - display: inline-flex; - align-items: center; - box-sizing: border-box; - vertical-align: middle; - justify-content: space-evenly; - } - - .jtoolbar-mobile > div { - display: flex; - width: 100%; - } - - .jtoolbar .jtoolbar-item { - text-align: center; - margin: auto; - padding: 2px; - padding-left: 4px; - padding-right: 4px; - } - - .jtoolbar-mobile .jtoolbar-item { - position: relative; - flex: 1; - } - - .jtoolbar .jtoolbar-divisor { - width: 2px; - height: 18px; - padding: 0px; - margin-left: 4px; - margin-right: 4px; - background-color: #ddd; - } - - .jtoolbar .jtoolbar-label { - padding-left: 8px; - padding-right: 8px; - } - - .jtoolbar-mobile a { - text-decoration: none; - display: inline-block; - } - - .jtoolbar-mobile i { - display: inline-flex !important; - color: #929292; - } - - .jtoolbar-mobile span { - font-size: 0.7em; - display: block; - color: #929292; - } - - .jtoolbar-mobile .jtoolbar-selected a, - .jtoolbar-mobile .jtoolbar-selected i, - .jtoolbar-mobile .jtoolbar-selected span { - color: var(--active-color) !important; - background-color: transparent; - } - - .jtoolbar-item { - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - } - - .jtoolbar-item i { - display: block; - color: #333; - } - - .jtoolbar-item:hover { - background-color: #f2f2f2; - } - - .jtoolbar .jpicker { - padding-left: 0px; - padding-right: 0px; - } - - .jtoolbar .jpicker-header { - height: 24px; - line-height: 24px; - padding: 0px; - padding-right: 20px; - padding-left: 8px; - background-position: top 50% right 0px; - display: flex; - align-items: center; - font-size: 0.9em; - } - - .jtoolbar .jpicker-content > div { - padding: 6px; - } - - .jtoolbar-active { - background-color: #eee; - } - - .jtoolbar .fa { - width: 18px; - height: 18px; - display: block; - line-height: 18px; - font-size: 14px; - } - - .jtoolbar .material-icons { - font-size: 18px; - width: 24px; - height: 24px; - display: block; - line-height: 24px; - transform: rotate(0.03deg); - text-align: center; - } - - .jtoolbar .jtoolbar-arrow { - background-repeat: no-repeat; - background-position: center; - background-image: url("data:image/svg+xml,%0A%3Csvg xmlns=%27http://www.w3.org/2000/svg%27 viewBox=%270 0 24 24%27 fill=%27black%27 width=%2718px%27 height=%2718px%27%3E%3Cpath d=%27M0 0h24v24H0z%27 fill=%27none%27/%3E%3Cpath d=%27M12 8c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm0 2c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm0 6c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z%27/%3E%3C/svg%3E"); - width: 24px; - height: 16px; - margin-left: 4px; - border-left: 1px solid #f2f2f2; - } - - .jtoolbar-floating { - position: absolute; - display: none; - 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); - border-radius: 4px; - background-color: #fff; - padding: 4px; - z-index: 50; - text-align: left; - margin-right: 20px; - } - - .jtoolbar-floating .jtoolbar-divisor { - display: none; - } - - .jtoolbar-arrow-selected .jtoolbar-floating { - display: flex; - flex-wrap: wrap; - } -} diff --git a/public/javascripts/jspreadsheet/jspreadsheet.js b/public/javascripts/jspreadsheet/jspreadsheet.js deleted file mode 100644 index 33ed309cd3b..00000000000 --- a/public/javascripts/jspreadsheet/jspreadsheet.js +++ /dev/null @@ -1,16905 +0,0 @@ -/** - * Jspreadsheet v4.11.1 - * - * Website: https://bossanova.uk/jspreadsheet/ - * Description: Create amazing web based spreadsheets. - * - * This software is distribute under MIT License - */ - -var formula = (function () { - // 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; - })(); - - var met = {}; - - met.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; - })(); - - met.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; - })(); - - met.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.95054210748189e31, - ], - ["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", 1e24, "Y"], - Z: ["zetta", 1e21, "Z"], - E: ["exa", 1e18, "E"], - P: ["peta", 1e15, "P"], - T: ["tera", 1e12, "T"], - G: ["giga", 1e9, "G"], - M: ["mega", 1e6, "M"], - k: ["kilo", 1e3, "k"], - h: ["hecto", 1e2, "h"], - e: ["dekao", 1e1, "e"], - d: ["deci", 1e-1, "d"], - c: ["centi", 1e-2, "c"], - m: ["milli", 1e-3, "m"], - u: ["micro", 1e-6, "u"], - n: ["nano", 1e-9, "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", 1e1, "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", 1e1, "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; - })(); - - met.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; - })(); - - met.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; - })(); - - met.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; - })(); - - met.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 () { - var args = utils.argsToArray(arguments); - var criteria = args.pop(); - var range = utils.parseNumberArray(utils.flatten(args)); - 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 += 2) { - if (isNaN(criteria[c][i])) { - condition += '"' + criteria[c][i] + '"' + criteria[c + 1]; - } else { - condition += criteria[c][i] + criteria[c + 1]; - } - if (c !== n_criterias - 1) { - condition += " && "; - } - } - condition = condition.slice(0, -4); - 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; - })(); - - met.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; - })(); - - met.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; - } - - if (text == null) { - return 0; - } - - 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; - })(); - - met.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 () { - var args = utils.argsToArray(arguments); - var criteria = args.pop(); - var range = utils.flatten(args); - 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; - })(); - - met.utils = (function () { - var exports = {}; - - exports.PROGRESS = function (p, c) { - var color = c ? c : "red"; - var value = p ? p : "0"; - - return ( - '
' - ); - }; - - exports.RATING = function (v) { - var html = ' "; - return html; - }; - - return exports; - })(); - - for (var i = 0; i < Object.keys(met).length; i++) { - var methods = met[Object.keys(met)[i]]; - var keys = Object.keys(methods); - for (var j = 0; j < keys.length; j++) { - if (!methods[keys[j]]) { - window[keys[j]] = function () { - return keys[j] + "Not implemented"; - }; - } else if ( - typeof methods[keys[j]] == "function" || - typeof methods[keys[j]] == "object" - ) { - window[keys[j]] = methods[keys[j]]; - window[keys[j]].toString = function () { - return "#ERROR"; - }; - - if (typeof methods[keys[j]] == "object") { - var tmp = Object.keys(methods[keys[j]]); - for (var z = 0; z < tmp.length; z++) { - window[keys[j]][tmp[z]].toString = function () { - return "#ERROR"; - }; - } - } - } else { - window[keys[j]] = function () { - return keys[j] + "Not implemented"; - }; - } - } - } - - /** - * Instance execution helpers - */ - var x = null; - var y = null; - var instance = null; - - window["TABLE"] = function () { - return instance; - }; - window["COLUMN"] = window["COL"] = function () { - return parseInt(x) + 1; - }; - window["ROW"] = function () { - return parseInt(y) + 1; - }; - window["CELL"] = function () { - return F.getColumnNameFromCoords(x, y); - }; - window["VALUE"] = function (col, row, processed) { - return instance.getValueFromCoords( - parseInt(col) - 1, - parseInt(row) - 1, - processed - ); - }; - window["THISROWCELL"] = function (col) { - return instance.getValueFromCoords(parseInt(col) - 1, parseInt(y)); - }; - - // Secure formula - var secureFormula = function (oldValue, runtime) { - var newValue = ""; - var inside = 0; - - var special = ["=", "!", ">", "<"]; - - 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(); - - if (runtime == true) { - if ( - i > 0 && - oldValue[i] == "=" && - special.indexOf(oldValue[i - 1]) == -1 && - special.indexOf(oldValue[i + 1]) == -1 - ) { - newValue += "="; - } - } - } - } - - // Adapt to JS - newValue = newValue.replace(/\^/g, "**"); - newValue = newValue.replace(/\<\>/g, "!="); - newValue = newValue.replace(/\&/g, "+"); - newValue = newValue.replace(/\$/g, ""); - - return newValue; - }; - - // Convert range tokens - var tokensUpdate = function (tokens, e) { - for (var index = 0; index < tokens.length; index++) { - var f = F.getTokensFromRange(tokens[index]); - e = e.replace(tokens[index], "[" + f.join(",") + "]"); - } - return e; - }; - - var F = function (expression, variables, i, j, obj) { - // Global helpers - instance = obj; - x = i; - y = j; - // String - var s = ""; - var keys = Object.keys(variables); - if (keys.length) { - for (var i = 0; i < keys.length; i++) { - if (keys[i].indexOf(".") == -1 && keys[i].indexOf("!") == -1) { - s += "var " + keys[i] + " = " + variables[keys[i]] + ";\n"; - } else { - s += keys[i] + " = " + variables[keys[i]] + ";\n"; - } - } - } - // Remove $ - expression = expression.replace(/\$/g, ""); - // Replace ! per dot - expression = expression.replace(/\!/g, "."); - // Adapt to JS - expression = secureFormula(expression, true); - // Update range - var tokens = expression.match( - /([A-Z]+[0-9]*\.)?(\$?[A-Z]+\$?[0-9]+):(\$?[A-Z]+\$?[0-9]+)?/g - ); - if (tokens && tokens.length) { - expression = tokensUpdate(tokens, expression); - } - - // Calculate - return new Function(s + "; return " + expression)(); - }; - - /** - * Get letter based on a number - * @param {number} i - * @return {string} - */ - var 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; - }; - - /** - * Get column name from coords - */ - F.getColumnNameFromCoords = function (x, y) { - return getColumnName(parseInt(x)) + (parseInt(y) + 1); - }; - - F.getCoordsFromColumnName = function (columnName) { - // Get the letters - var t = /^[a-zA-Z]+/.exec(columnName); - - 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 jspreadsheet starts on zero - if (code < 0) { - code = 0; - } - - // Number - var number = parseInt(/[0-9]+$/.exec(columnName)) || null; - if (number > 0) { - number--; - } - - return [code, number]; - } - }; - - F.getRangeFromTokens = function (tokens) { - tokens = tokens.filter(function (v) { - return v != "#REF!"; - }); - - var d = ""; - var t = ""; - for (var i = 0; i < tokens.length; i++) { - if (tokens[i].indexOf(".") >= 0) { - d = "."; - } else if (tokens[i].indexOf("!") >= 0) { - d = "!"; - } - if (d) { - t = tokens[i].split(d); - tokens[i] = t[1]; - t = t[0] + d; - } - } - - tokens.sort(function (a, b) { - var t1 = Helpers.getCoordsFromColumnName(a); - var t2 = Helpers.getCoordsFromColumnName(b); - if (t1[1] > t2[1]) { - return 1; - } else if (t1[1] < t2[1]) { - return -1; - } else { - if (t1[0] > t2[0]) { - return 1; - } else if (t1[0] < t2[0]) { - return -1; - } else { - return 0; - } - } - }); - - if (!tokens.length) { - return "#REF!"; - } else { - return t + (tokens[0] + ":" + tokens[tokens.length - 1]); - } - }; - - F.getTokensFromRange = function (range) { - if (range.indexOf(".") > 0) { - var t = range.split("."); - range = t[1]; - t = t[0] + "."; - } else if (range.indexOf("!") > 0) { - var t = range.split("!"); - range = t[1]; - t = t[0] + "!"; - } else { - var t = ""; - } - - var range = range.split(":"); - var e1 = F.getCoordsFromColumnName(range[0]); - var e2 = F.getCoordsFromColumnName(range[1]); - - 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] === null && e2[1] == null) { - var y1 = null; - var y2 = null; - - var k = Object.keys(vars); - for (var i = 0; i < k.length; i++) { - var tmp = F.getCoordsFromColumnName(k[i]); - if (tmp[0] === e1[0]) { - if (y1 === null || tmp[1] < y1) { - y1 = tmp[1]; - } - } - if (tmp[0] === e2[0]) { - if (y2 === null || tmp[1] > y2) { - y2 = tmp[1]; - } - } - } - } else { - if (e1[1] <= e2[1]) { - var y1 = e1[1]; - var y2 = e2[1]; - } else { - var y1 = e2[1]; - var y2 = e1[1]; - } - } - - var f = []; - for (var j = y1; j <= y2; j++) { - var line = []; - for (var i = x1; i <= x2; i++) { - line.push(t + F.getColumnNameFromCoords(i, j)); - } - f.push(line); - } - - return f; - }; - - F.setFormula = function (o) { - var k = Object.keys(o); - for (var i = 0; i < k.length; i++) { - if (typeof o[k[i]] == "function") { - window[k[i]] = o[k[i]]; - } - } - }; - - return F; -})(); - -if (!jSuites && typeof require === "function") { - var jSuites = require("jsuites"); -} - -(function (global, factory) { - typeof exports === "object" && typeof module !== "undefined" - ? (module.exports = factory()) - : typeof define === "function" && define.amd - ? define(factory) - : (global.jspreadsheet = global.jexcel = factory()); -})(this, function () { - "use strict"; - - // Basic version information - var Version = (function () { - // Information - var info = { - title: "Jspreadsheet", - version: "4.11.1", - type: "CE", - host: "https://bossanova.uk/jspreadsheet", - license: "MIT", - print: function () { - return [ - this.title + " " + this.type + " " + this.version, - this.host, - this.license, - ].join("\r\n"); - }, - }; - - return function () { - return info; - }; - })(); - - /** - * The value is a formula - */ - var isFormula = function (value) { - var v = ("" + value)[0]; - return v == "=" || v == "#" ? true : false; - }; - - /** - * Get the mask in the jSuites.mask format - */ - var getMask = function (o) { - if (o.format || o.mask || o.locale) { - var opt = {}; - if (o.mask) { - opt.mask = o.mask; - } else if (o.format) { - opt.mask = o.format; - } else { - opt.locale = o.locale; - opt.options = o.options; - } - - if (o.decimal) { - if (!opt.options) { - opt.options = {}; - } - opt.options = { decimal: o.decimal }; - } - return opt; - } - - return null; - }; - - // Jspreadsheet core object - var jexcel = function (el, options) { - // Create jspreadsheet object - var obj = {}; - obj.options = {}; - - if (!(el instanceof Element || el instanceof HTMLDocument)) { - console.error("Jspreadsheet: 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("Jspreadsheet: el is not a valid DOM element"); - return false; - } - } - - // Loading default configuration - var defaults = { - // External data - url: null, - // Ajax options - method: "GET", - requestVariables: null, - // Data - data: null, - // Custom sorting handler - sorting: 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", - // Rows height default - defaultRowHeight: null, - // 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: "jspreadsheet", - // 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, - textOverflow: false, - // Meta - meta: null, - // Style - style: null, - classes: null, - // Execute formulas - parseFormulas: true, - autoIncrement: true, - autoCasting: true, - // Security - secureFormulas: true, - stripHTML: true, - stripHTMLOnCopy: false, - // Filters - filters: false, - footers: null, - // Event handles - onundo: null, - onredo: null, - onload: null, - onchange: null, - oncomments: 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, - oncreateeditor: 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: true, - }; - - // 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( - "Jspreadsheet: 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 jspreadsheet 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({ - url: obj.options.columns[i].url, - index: i, - method: "GET", - dataType: "json", - success: function (data) { - var source = []; - for (var i = 0; i < data.length; i++) { - obj.options.columns[this.index].source.push(data[i]); - } - }, - }); - } - } 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"; - } - } - } - // Create the table when is ready - if (!multiple.length) { - obj.createTable(); - } else { - jSuites.ajax(multiple, function () { - 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); - } - - // Set initial pagination value - obj.paginationDropdown.value = obj.options.pagination; - - 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 = " "; - td.setAttribute("data-x", i); - td.className = "jexcel_column_filter"; - if (obj.options.columns[i].type == "hidden") { - td.style.display = "none"; - } - 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); - - if (!obj.options.textOverflow) { - obj.table.classList.add("jexcel_overflow"); - } - - // 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 Jspreadsheet - var ads = document.createElement("a"); - ads.setAttribute("href", "https://bossanova.uk/jspreadsheet/"); - 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/jspreadsheet/logo.png"; - ads.appendChild(img); - } - } catch (exception) {} - var span = document.createElement("span"); - span.innerHTML = "Jspreadsheet CE"; - 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["box-shadow"] = - "rgb(221 221 221) 2px 2px 5px 0.1px"; - 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); - } - - // Classes - if (obj.options.classes) { - var k = Object.keys(obj.options.classes); - for (var i = 0; i < k.length; i++) { - var cell = jexcel.getIdFromColumnName(k[i], true); - obj.records[cell[1]][cell[0]].classList.add( - obj.options.classes[k[i]] - ); - } - } - }; - - /** - * 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: obj.options.method, - data: obj.options.requestVariables, - 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( - "Jspreadsheet: 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 json data by row number - * - * @param integer row number - * @return object - */ - obj.getJsonRow = function (rowNumber) { - var rowData = obj.options.data[rowNumber]; - var x = obj.options.columns.length; - - var row = {}; - for (var i = 0; i < x; i++) { - if (!obj.options.columns[i].name) { - obj.options.columns[i].name = i; - } - row[obj.options.columns[i].name] = rowData[i]; - } - - return row; - }; - - /** - * Get the whole table data - * - * @param bool highlighted cells only - * @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); - 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); - // Index - var index = null; - - // Set default row height - if (obj.options.defaultRowHeight) { - obj.rows[j].style.height = obj.options.defaultRowHeight + "px"; - } - - // Definitions - if (obj.options.rows[j]) { - if (obj.options.rows[j].height) { - obj.rows[j].style.height = obj.options.rows[j].height; - } - if (obj.options.rows[j].title) { - index = obj.options.rows[j].title; - } - } - if (!index) { - index = parseInt(j + 1); - } - // Row number label - var td = document.createElement("td"); - td.innerHTML = index; - 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, cell) { - if ( - ("" + value).substr(0, 1) == "=" && - obj.options.parseFormulas == true - ) { - value = obj.executeFormula(value, i, j); - } - - // Column options - var options = obj.options.columns[i]; - if (options && !isFormula(value)) { - // Mask options - var opt = null; - if ((opt = getMask(options))) { - if (value && value == Number(value)) { - value = Number(value); - } - // Process the decimals to match the mask - var masked = jSuites.mask.render(value, opt, true); - // Negative indication - if (cell) { - if (opt.mask) { - var t = opt.mask.split(";"); - if (t[1]) { - var t1 = t[1].match(new RegExp("\\[Red\\]", "gi")); - if (t1) { - if (value < 0) { - cell.classList.add("red"); - } else { - cell.classList.remove("red"); - } - } - var t2 = t[1].match(new RegExp("\\(", "gi")); - if (t2) { - if (value < 0) { - masked = "(" + masked + ")"; - } - } - } - } - } - - if (masked) { - value = masked; - } - } - } - - return value; - }; - - var validDate = function (date) { - date = "" + date; - if (date.substr(4, 1) == "-" && date.substr(7, 1) == "-") { - return true; - } else { - date = date.split("-"); - if ( - date[0].length == 4 && - date[0] == Number(date[0]) && - date[1].length == 2 && - date[1] == Number(date[1]) - ) { - return true; - } - } - return false; - }; - - /** - * 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.textContent = 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.textContent = 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 = null; - if (!validDate(value)) { - var tmp = jSuites.calendar.extractDateFromString( - value, - obj.options.columns[i].options.format - ); - if (tmp) { - formatted = tmp; - } - } - // Create calendar cell - td.textContent = 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.textContent = 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.textContent = 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, td)); - } else { - if ( - obj.options.stripHTML === false || - obj.options.columns[i].stripHTML === false - ) { - td.innerHTML = stripScript(obj.parseValue(i, j, value, td)); - } else { - td.textContent = obj.parseValue(i, j, value, td); - } - } - } - } - - // 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 (this.options.textOverflow == true) { - 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"); - if (obj.options.stripHTML) { - obj.headers[colNumber].textContent = obj.options.columns[colNumber] - .title - ? obj.options.columns[colNumber].title - : jexcel.getColumnName(colNumber); - } else { - obj.headers[colNumber].innerHTML = 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 - ); - } - if (obj.options.columns[colNumber].id) { - obj.headers[colNumber].setAttribute( - "id", - obj.options.columns[colNumber].id - ); - } - - // 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].textContent = - 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.textContent = 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); - toolbarItem.setAttribute("id", toolbar[i].id); - - // 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.textContent = 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.textContent = 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.textContent = 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("Jspreadsheet: filters not enabled."); - } else { - // Make sure is integer - columnId = parseInt(columnId); - // Reset selection - obj.resetSelection(); - // Load options - var optionsFiltered = []; - if (obj.options.columns[columnId].type == "checkbox") { - optionsFiltered.push({ id: "true", name: "True" }); - optionsFiltered.push({ id: "false", name: "False" }); - } else { - var options = []; - var hasBlanks = false; - 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; - } else { - var hasBlanks = true; - } - } - var keys = Object.keys(options); - var optionsFiltered = []; - for (var j = 0; j < keys.length; j++) { - optionsFiltered.push({ id: keys[j], name: options[keys[j]] }); - } - // Has blank options - if (hasBlanks) { - optionsFiltered.push({ value: "", id: "", name: "(Blanks)" }); - } - } - - // 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"; - - var opt = { - data: optionsFiltered, - multiple: true, - autocomplete: true, - opened: true, - value: - obj.filters[columnId] !== undefined ? obj.filters[columnId] : null, - 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.refreshSelection(); - }, - }; - - // Dynamic dropdown - jSuites.dropdown(div, opt); - } - }; - - obj.resetFilters = function () { - if (obj.options.filters) { - for (var i = 0; i < obj.filter.children.length; i++) { - obj.filter.children[i].innerHTML = " "; - obj.filters[i] = null; - } - } - - obj.results = null; - obj.updateResult(); - }; - - 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++) { - var value = "" + obj.options.data[y][x]; - var label = "" + obj.records[y][x].innerHTML; - if (query[i] == value || query[i] == label) { - 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); - - // On edition start - obj.dispatch("oncreateeditor", el, cell, x, y, 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]; - if (obj.options.columns[x].multiple && !Array.isArray(value)) { - value = value.split(";"); - } - - // 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; - } - - // Do not change the original source - var data = []; - for (var j = 0; j < source.length; j++) { - data.push(source[j]); - } - - // Create editor - var editor = createEditor("div"); - var options = { - data: data, - 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: 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, - }); - var rect = cell.getBoundingClientRect(); - var 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); - var rect = cell.getBoundingClientRect(); - var 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"); - } - - editor.focus(); - editor.value = value; - - // Column options - var options = obj.options.columns[x]; - // Format - var opt = null; - - // Apply format when is not a formula - if (!isFormula(value)) { - // Format - if ((opt = getMask(options))) { - // Masking - if (!options.disabledMaskOnEdition) { - if (options.mask) { - var m = options.mask.split(";"); - editor.setAttribute("data-mask", m[0]); - } else if (options.locale) { - editor.setAttribute("data-locale", options.locale); - } - } - // Input - opt.input = editor; - // Configuration - editor.mask = opt; - // Do not treat the decimals - jSuites.mask.render(value, opt, false); - } - } - - editor.onblur = function () { - obj.closeEditor(cell, true); - }; - editor.scrollLeft = editor.scrollWidth; - } - } - } - }; - - /** - * 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; - - // Column options - var options = obj.options.columns[x]; - // Format - var opt = null; - if ((opt = getMask(options))) { - // Keep numeric in the raw data - if ( - value !== "" && - !isFormula(value) && - typeof value !== "number" - ) { - var t = jSuites.mask.extract(value, opt, true); - if (t && t.value !== "") { - value = t.value; - } - } - } - } - } - - // 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 column options - * @param x - * @param y - * @returns {{type: string}} - */ - obj.getColumnOptions = function (x, y) { - // Type - var options = obj.options.columns[x]; - - // Cell type - if (!options) { - options = { type: "text" }; - } - - return options; - }; - - /** - * 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].textContent = obj.getDropDownValue(x, value); - } else if (obj.options.columns[x].type == "calendar") { - // Try formatted date - var formatted = null; - if (!validDate(value)) { - var tmp = jSuites.calendar.extractDateFromString( - value, - obj.options.columns[x].options.format - ); - if (tmp) { - formatted = tmp; - } - } - // Update data and cell - obj.options.data[y][x] = value; - obj.records[y][x].textContent = 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].textContent = ""; - obj.records[y][x].appendChild(color); - } else { - obj.records[y][x].style.color = value; - obj.records[y][x].textContent = 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, obj.records[y][x]) - ); - } else { - obj.records[y][x].textContent = obj.parseValue( - x, - y, - value, - obj.records[y][x] - ); - } - } - // 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 overlaps 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 highlighted 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(); - - // select column - if (y1 == null) { - y1 = 0; - y2 = obj.rows.length - 1; - } - - // 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] && obj.records[borderTop][i]) { - obj.records[borderTop][i].classList.add("highlight-top"); - } - // Bottom border - if (obj.records[borderBottom] && 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] && 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]; - var lastX = last.getAttribute("data-x"); - - var contentRect = obj.content.getBoundingClientRect(); - var x1 = contentRect.left; - var y1 = contentRect.top; - - var 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(); - // Only check if the last column is not part of the merged cells - if (lastX > obj.options.freezeColumns - 1 && x2 - x1 + w2 < width) { - obj.corner.style.display = "none"; - } else { - if (obj.options.selectionCopy == true) { - obj.corner.style.display = ""; - } - } - } else { - if (obj.options.selectionCopy == true) { - obj.corner.style.display = ""; - } - } - } - }; - - /** - * Update scroll position based on the selection - */ - obj.updateScroll = function (direction) { - // Jspreadsheet Container information - var 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 - var 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 (typeof column === "undefined") { - // 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; - - // 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 (typeof row === "undefined") { - // 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.textContent = obj.parseValue( - +obj.records.length + i, - j, - obj.options.footers[j][i] - ); - - // Hide/Show with hideColumn()/showColumn() - td.style.display = obj.colgroup[i].style.display; - } - } - } - }; - - /** - * 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].textContent; - }; - - /** - * 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].textContent; - - if (!newValue) { - newValue = prompt(obj.options.text.columnName, oldValue); - } - - if (newValue) { - obj.headers[column].textContent = 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]] && - (obj.records[cell[1]][cell[0]].classList.contains("readonly") == - false || - force) - ) { - // 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, - }); - // Set comments - obj.dispatch("oncomments", el, comments, title, cell, cell[0], cell[1]); - }; - - /** - * 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; - } - - // Test order - var temp = []; - if ( - obj.options.columns[column].type == "number" || - obj.options.columns[column].type == "numeric" || - 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].textContent.toLowerCase()]; - } - } - - // Default sorting method - if (typeof obj.options.sorting !== "function") { - obj.options.sorting = function (direction) { - return function (a, b) { - var valueA = a[1]; - var valueB = b[1]; - - if (!direction) { - 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; - } - }; - }; - } - - temp = temp.sort(obj.options.sorting(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 - ) { - 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 records - if (rowNumber + numOfRows >= obj.options.data.length) { - numOfRows = obj.options.data.length - rowNumber; - } - - // Onbeforedeleterow - if ( - obj.dispatch("onbeforedeleterow", el, rowNumber, numOfRows) === - 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; - } - - // If delete all rows, and set allowDeletingAllRows false, will stay one row - if ( - obj.options.allowDeletingAllRows == false && - lastRow + 1 === numOfRows - ) { - numOfRows--; - console.error( - "Jspreadsheet: It is not possible to delete the last row" - ); - } - - // 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( - "Jspreadsheet: 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 - ) { - 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); - var o = - obj.thead.children[j].children[ - obj.thead.children[j].children.length - 1 - ].getAttribute("data-column"); - o = o.split(","); - for ( - var col = columnIndex; - col < numOfColumns + columnIndex; - col++ - ) { - o.push(col); - } - obj.thead.children[j].children[ - obj.thead.children[j].children.length - 1 - ].setAttribute("data-column", o); - } - } 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 - ); - } - }; - - /** - * 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 - ) { - 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( - "Jspreadsheet: It is not possible to delete the last column" - ); - } - } - }; - - /** - * Get selected 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 selected 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].textContent, - 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); - }; - - /** - * Readonly - */ - obj.isReadOnly = function (cell) { - if ((cell = obj.getCell(cell))) { - return cell.classList.contains("readonly") ? true : false; - } - }; - - /** - * Readonly - */ - obj.setReadOnly = function (cell, state) { - if ((cell = obj.getCell(cell))) { - if (state) { - cell.classList.add("readonly"); - } else { - cell.classList.remove("readonly"); - } - } - }; - - /** - * 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 = ""; - if (obj.filter && obj.filter.children.length > colNumber + 1) { - obj.filter.children[colNumber + 1].style.display = ""; - } - for (var j = 0; j < obj.options.data.length; j++) { - obj.records[j][colNumber].style.display = ""; - } - - // Update footers - if (obj.options.footers) { - obj.setFooter(); - } - - obj.resetSelection(); - }; - - /** - * Hide column - */ - obj.hideColumn = function (colNumber) { - obj.headers[colNumber].style.display = "none"; - obj.colgroup[colNumber].style.display = "none"; - if (obj.filter && obj.filter.children.length > colNumber + 1) { - obj.filter.children[colNumber + 1].style.display = "none"; - } - for (var j = 0; j < obj.options.data.length; j++) { - obj.records[j][colNumber].style.display = "none"; - } - - // Update footers - if (obj.options.footers) { - obj.setFooter(); - } - - obj.resetSelection(); - }; - - /** - * 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(",")); - } - }; - - // Range with $ remove $ - expression = expression.replace(/\$?([A-Z]+)\$?([0-9]+)/g, "$1$2"); - - var tokens = expression.match(/([A-Z]+[0-9]+)\:([A-Z]+[0-9]+)/g); - if (tokens && tokens.length) { - tokensUpdate(tokens); - } - - // 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 { - // Expressions to be used in the parsing - var formulaExpressions = {}; - - 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 - formulaExpressions[tokens[i]] = null; - } else { - if ( - value == Number(value) && - obj.options.autoCasting == true - ) { - // Number - formulaExpressions[tokens[i]] = Number(value); - } else { - // Trying any formatted number - var number = obj.parseNumber(value, position[0]); - if (obj.options.autoCasting == true && number) { - formulaExpressions[tokens[i]] = number; - } else { - formulaExpressions[tokens[i]] = '"' + value + '"'; - } - } - } - } - } - } - - // Convert formula to javascript - try { - var res = jexcel.formula( - expression.substr(1), - formulaExpressions, - x, - y, - obj - ); - } catch (e) { - var res = "#ERROR"; - console.log(e); - } - - 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.isInteger(Number(number[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.options.filters == 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) - 1; - } - - 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.options.filters == 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.options.filters == 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.options.filters == 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.options.filters == 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.options.filters == 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.options.filters == 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.options.filters == 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.options.filters == 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.options.filters == 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) - 1; - } - - // 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.options.filters == 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.options.filters == 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 = ""; - - // Get data - data += obj.copy( - false, - obj.options.csvDelimiter, - true, - includeHeaders, - 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, - obj.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, - includeHeaders, - download - ) { - if (!delimiter) { - delimiter = "\t"; - } - - var div = new RegExp(delimiter, "ig"); - - // Controls - var header = []; - var col = []; - var colLabel = []; - var row = []; - var rowLabel = []; - var x = obj.options.data[0].length; - var y = obj.options.data.length; - var tmp = ""; - var copyHeader = false; - var headers = ""; - var nestedHeaders = ""; - var numOfCols = 0; - var numOfRows = 0; - - // Partial copy - var copyX = 0; - var copyY = 0; - var isPartialCopy = true; - // Go through the columns to get the data - for (var j = 0; j < y; j++) { - for (var i = 0; i < x; i++) { - // If cell is highlighted - if ( - !highlighted || - obj.records[j][i].classList.contains("highlight") - ) { - if (copyX <= i) { - copyX = i; - } - if (copyY <= j) { - copyY = j; - } - } - } - } - if (x === copyX + 1 && y === copyY + 1) { - isPartialCopy = false; - } - - if ( - (download && obj.options.includeHeadersOnDownload == true) || - (!download && - obj.options.includeHeadersOnCopy == true && - !isPartialCopy) || - includeHeaders - ) { - // 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]) - ) { - tmp = [obj.options.nestedHeaders]; - } else { - tmp = obj.options.nestedHeaders; - } - - for (var j = 0; j < tmp.length; j++) { - var nested = []; - for (var i = 0; i < tmp[j].length; i++) { - var colspan = parseInt(tmp[j][i].colspan); - nested.push(tmp[j][i].title); - for (var c = 0; c < colspan - 1; c++) { - nested.push(""); - } - } - nestedHeaders += nested.join(delimiter) + "\r\n"; - } - } - - copyHeader = true; - } - - // 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") - ) { - if (copyHeader == true) { - header.push(obj.headers[i].textContent); - } - // Values - var value = obj.options.data[j][i]; - if ( - value.match && - (value.match(div) || - 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 { - if (obj.options.stripHTMLOnCopy == true) { - var label = obj.records[j][i].textContent; - } else { - var label = obj.records[j][i].innerHTML; - } - if ( - label.match && - (label.match(div) || - 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) { - if (copyHeader) { - numOfCols = col.length; - row.push(header.join(delimiter)); - } - row.push(col.join(delimiter)); - } - if (colLabel.length) { - numOfRows++; - if (copyHeader) { - rowLabel.push(header.join(delimiter)); - copyHeader = false; - } - rowLabel.push(colLabel.join(delimiter)); - } - } - - if (x == numOfCols && y == numOfRows) { - headers = nestedHeaders; - } - - // Final string - var str = headers + row.join("\r\n"); - var strLabel = headers + 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 - if (!returnData) { - 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; - }; - - /** - * Jspreadsheet 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) { - // If the pasted column is out of range, create it if possible - if (obj.options.allowInsertColumn == true) { - obj.insertColumn(); - // Otherwise skip the pasted data that overflows - } else { - break; - } - } - colIndex = obj.right.get(colIndex, rowIndex); - } - } - - j++; - if (data[j]) { - if (rowIndex >= obj.rows.length - 1) { - // If the pasted row is out of range, create it if possible - if (obj.options.allowInsertRow == true) { - obj.insertRow(); - // Otherwise skip the pasted data that overflows - } else { - break; - } - } - 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 == "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 == "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]; - } - } - - // Guarantee single multiple compatibility - var keys = Array.isArray(key) ? key : ("" + key).split(";"); - - for (var i = 0; i < keys.length; i++) { - if (typeof keys[i] === "object") { - value.push(combo[keys[i].id]); - } else { - if (combo[keys[i]]) { - value.push(combo[keys[i]]); - } - } - } - } else { - console.error("Invalid column"); - } - - return value.length > 0 ? value.join("; ") : ""; - }; - - /** - * From stack overflow 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: obj.options.method, - data: obj.options.requestVariables, - 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: obj.options.method, - data: obj.options.requestVariables, - 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 () { - if (obj.options.about === true) { - alert(Version().print()); - } else { - 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 - 10 - ) { - 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) { - // Must check if the previous column is hidden or not to determin whether the width shoule be added or not! - if (obj.options.columns[i - 1].type !== "hidden") { - 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; - el.jspreadsheet = obj; - - obj.init(); - - return obj; - }; - - // Define dictionary - jexcel.setDictionary = function (o) { - jSuites.setDictionary(o); - }; - - // Define extensions - jexcel.setExtensions = function (o) { - var k = Object.keys(o); - for (var i = 0; i < k.length; i++) { - if (typeof o[k[i]] === "function") { - jexcel[k[i]] = o[k[i]]; - if (jexcel.license && typeof o[k[i]].license == "function") { - o[k[i]].license(jexcel.license); - } - } - } - }; - - /** - * Formulas - */ - if (typeof formula !== "undefined") { - jexcel.formula = formula; - } - jexcel.version = Version; - - jexcel.current = null; - jexcel.timeControl = null; - jexcel.timeControlLoading = null; - - const destroyEvents = function (root) { - 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.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) { - destroyEvents(root); - jexcel = null; - } - } - }; - - jexcel.build = function (root) { - destroyEvents(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 ( - ["calendar", "html"].includes( - jexcel.current.options.columns[jexcel.current.edition[2]].type - ) - ) { - 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 == 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 == 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 - e.preventDefault(); - if ( - jspreadsheet.current.options.columns[columnId].type == - "checkbox" || - jspreadsheet.current.options.columns[columnId].type == - "radio" - ) { - jspreadsheet.current.setCheckRadioValue(); - } else { - // Start edition - jspreadsheet.current.openEditor( - jspreadsheet.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 <= 190) || - ((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( - "Jspreadsheet: 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( - "Jspreadsheet: This row is part of a merged cell" - ); - } else if ( - jexcel.current.options.search == true && - jexcel.current.results - ) { - console.error( - "Jspreadsheet: 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.textContent == "<") { - jexcel.current.page(0); - } else if (e.target.textContent == ">") { - jexcel.current.page(e.target.getAttribute("title") - 1); - } else { - jexcel.current.page(e.target.textContent - 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 != jexcel.current.dragging.destination - ) { - 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 if (jexcel.current.dragging) { - if (jexcel.current.dragging.column) { - var columnId = e.target.getAttribute("data-x"); - if (columnId) { - if (jexcel.current.isColMerged(columnId).length) { - console.error( - "Jspreadsheet: 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 { - var rowId = e.target.getAttribute("data-y"); - if (rowId) { - if (jexcel.current.isRowMerged(rowId).length) { - console.error( - "Jspreadsheet: This row is part of a merged cell." - ); - } else { - var target = - e.target.clientHeight / 2 > e.offsetY - ? e.target.parentNode.nextSibling - : e.target.parentNode; - if (jexcel.current.dragging.element != target) { - e.target.parentNode.parentNode.insertBefore( - jexcel.current.dragging.element, - target - ); - jexcel.current.dragging.destination = - Array.prototype.indexOf.call( - jexcel.current.dragging.element.parentNode.children, - jexcel.current.dragging.element - ); - } - } - } - } - } - } 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.resizing || jexcel.current.dragging) { - } 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) { - if ( - x < parseInt(jexcel.current.selectedCell[0]) || - x > parseInt(jexcel.current.selectedCell[2]) || - y < parseInt(jexcel.current.selectedCell[1]) || - y > parseInt(jexcel.current.selectedCell[3]) - ) { - jexcel.current.updateSelectionFromCoords(x, y, 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.innerHTML; - options.columns[i].align = header.style.textAlign || "center"; - - if ((info = header.getAttribute("name"))) { - options.columns[i].name = info; - } - if ((info = header.getAttribute("id"))) { - options.columns[i].id = info; - } - }; - - // Headers - var nested = []; - var headers = el.querySelectorAll(":scope > thead > tr"); - if (headers.length) { - for (var j = 0; j < headers.length - 1; j++) { - var cells = []; - for (var i = 0; i < headers[j].children.length; i++) { - var row = { - title: headers[j].children[i].textContent, - colspan: headers[j].children[i].getAttribute("colspan") || 1, - }; - cells.push(row); - } - nested.push(cells); - } - // 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 classes = {}; - - var content = el.querySelectorAll(":scope > tr, :scope > tbody > tr"); - for (var j = 0; j < content.length; j++) { - options.data[rowNumber] = []; - if ( - options.parseTableFirstRowAsHeader == true && - !headers.length && - 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("data-formula"); - if (value) { - if (value.substr(0, 1) != "=") { - value = "=" + value; - } - } else { - var value = content[j].children[i].innerHTML; - } - options.data[rowNumber].push(value); - - // Key - var cellName = jexcel.getColumnNameFromId([i, j]); - - // Classes - var tmp = content[j].children[i].getAttribute("class"); - if (tmp) { - classes[cellName] = tmp; - } - - // 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")) { - if (style[cellName]) { - style[cellName] += "; font-weight:bold;"; - } else { - style[cellName] = "font-weight:bold;"; - } - } - } - - // Row Height - if (content[j].style && content[j].style.height) { - rows[j] = { height: content[j].style.height }; - } - - // Index - rowNumber++; - } - } - - // Nested - if (Object.keys(nested).length > 0) { - options.nestedHeaders = nested; - } - // 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; - } - // Classes - if (Object.keys(classes).length > 0) { - options.classes = classes; - } - - var content = el.querySelectorAll("tfoot tr"); - if (content.length) { - var footers = []; - for (var j = 0; j < content.length; j++) { - var footer = []; - for (var i = 0; i < content[j].children.length; i++) { - footer.push(content[j].children[i].textContent); - } - footers.push(footer); - } - if (Object.keys(footers).length > 0) { - options.footers = footers; - } - } - // 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; - } - }; - - // Helpers - jexcel.helpers = (function () { - var component = {}; - - /** - * Get carret position for one element - */ - component.getCaretIndex = function (e) { - if (this.config.root) { - var d = this.config.root; - } else { - var d = window; - } - var pos = 0; - var s = d.getSelection(); - if (s) { - if (s.rangeCount !== 0) { - var r = s.getRangeAt(0); - var p = r.cloneRange(); - p.selectNodeContents(e); - p.setEnd(r.endContainer, r.endOffset); - pos = p.toString().length; - } - } - return pos; - }; - - /** - * Invert keys and values - */ - component.invert = function (o) { - var d = []; - var k = Object.keys(o); - for (var i = 0; i < k.length; i++) { - d[o[k[i]]] = k[i]; - } - return d; - }; - - /** - * Get letter based on a number - * - * @param integer i - * @return string letter - */ - component.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; - }; - - /** - * Get column name from coords - */ - component.getColumnNameFromCoords = function (x, y) { - return component.getColumnName(parseInt(x)) + (parseInt(y) + 1); - }; - - component.getCoordsFromColumnName = function (columnName) { - // Get the letters - var t = /^[a-zA-Z]+/.exec(columnName); - - 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 jspreadsheet starts on zero - if (code < 0) { - code = 0; - } - - // Number - var number = parseInt(/[0-9]+$/.exec(columnName)) || null; - if (number > 0) { - number--; - } - - return [code, number]; - } - }; - - /** - * Extract json configuration from a TABLE DOM tag - */ - component.createFromTable = function () {}; - - /** - * Helper injectArray - */ - component.injectArray = function (o, idx, arr) { - return o.slice(0, idx).concat(arr).concat(o.slice(idx)); - }; - - /** - * Parse CSV string to JS array - */ - component.parseCSV = function (str, delimiter) { - // user-supplied delimeter or default comma - delimiter = delimiter || ","; - - // Final data - var col = 0; - var row = 0; - var num = 0; - var data = [[]]; - var limit = 0; - var flag = null; - var inside = false; - var closed = false; - - // Go over all chars - for (var i = 0; i < str.length; i++) { - // Create new row - if (!data[row]) { - data[row] = []; - } - // Create new column - if (!data[row][col]) { - data[row][col] = ""; - } - - // Ignore - if (str[i] == "\r") { - continue; - } - - // New row - if ( - (str[i] == "\n" || str[i] == delimiter) && - (inside == false || closed == true || !flag) - ) { - // Restart flags - flag = null; - inside = false; - closed = false; - - if (data[row][col][0] == '"') { - var val = data[row][col].trim(); - if (val[val.length - 1] == '"') { - data[row][col] = val.substr(1, val.length - 2); - } - } - - // Go to the next cell - if (str[i] == "\n") { - // New line - col = 0; - row++; - } else { - // New column - col++; - if (col > limit) { - // Keep the reference of max column - limit = col; - } - } - } else { - // Inside quotes - if (str[i] == '"') { - inside = !inside; - } - - if (flag === null) { - flag = inside; - if (flag == true) { - continue; - } - } else if (flag === true && !closed) { - if (str[i] == '"') { - if (str[i + 1] == '"') { - inside = true; - data[row][col] += str[i]; - i++; - } else { - closed = true; - } - continue; - } - } - - data[row][col] += str[i]; - } - } - - // Make sure a square matrix is generated - for (var j = 0; j < data.length; j++) { - for (var i = 0; i <= limit; i++) { - if (data[j][i] === undefined) { - data[j][i] = ""; - } - } - } - - return data; - }; - - return component; - })(); - - /** - * Jquery Support - */ - if (typeof jQuery != "undefined") { - (function ($) { - $.fn.jspreadsheet = $.fn.jexcel = function (mixed) { - var spreadsheetContainer = $(this).get(0); - if (!spreadsheetContainer.jexcel) { - return jexcel($(this).get(0), arguments[0]); - } else { - if (Array.isArray(spreadsheetContainer.jexcel)) { - return spreadsheetContainer.jexcel[mixed][arguments[1]].apply( - this, - Array.prototype.slice.call(arguments, 2) - ); - } else { - return spreadsheetContainer.jexcel[mixed].apply( - this, - Array.prototype.slice.call(arguments, 1) - ); - } - } - }; - })(jQuery); - } - - return jexcel; -}); diff --git a/public/javascripts/jsuites/jsuites.js b/public/javascripts/jsuites/jsuites.js deleted file mode 100644 index 0088464e5aa..00000000000 --- a/public/javascripts/jsuites/jsuites.js +++ /dev/null @@ -1,12414 +0,0 @@ -/** - * (c) jSuites Javascript Web Components - * - * Website: https://jsuites.net - * 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 = {}; - -var Version = '4.17.5'; - -var Events = function() { - - document.jsuitesComponents = []; - - var find = function(DOMElement, component) { - if (DOMElement[component.type] && DOMElement[component.type] == component) { - return true; - } - if (DOMElement.component && DOMElement.component == component) { - return true; - } - if (DOMElement.parentNode) { - return find(DOMElement.parentNode, component); - } - return false; - } - - var isOpened = function(e) { - if (document.jsuitesComponents && document.jsuitesComponents.length > 0) { - for (var i = 0; i < document.jsuitesComponents.length; i++) { - if (document.jsuitesComponents[i] && ! find(e, document.jsuitesComponents[i])) { - document.jsuitesComponents[i].close(); - } - } - } - } - - // Width of the border - var cornerSize = 15; - - // Current element - var element = null; - - // Controllers - var editorAction = false; - - // Event state - var state = { - x: null, - y: null, - } - - // Tooltip element - var tooltip = document.createElement('div') - tooltip.classList.add('jtooltip'); - - // Events - var mouseDown = function(e) { - // Check if this is the floating - var item = jSuites.findElement(e.target, 'jpanel'); - // Jfloating found - if (item && ! item.classList.contains("readonly")) { - // Add focus to the chart container - item.focus(); - // Keep the tracking information - var rect = e.target.getBoundingClientRect(); - editorAction = { - e: item, - x: e.clientX, - y: e.clientY, - w: rect.width, - h: rect.height, - d: item.style.cursor, - resizing: item.style.cursor ? true : false, - actioned: false, - } - - // Make sure width and height styling is OK - if (! item.style.width) { - item.style.width = rect.width + 'px'; - } - - if (! item.style.height) { - item.style.height = rect.height + 'px'; - } - - // Remove any selection from the page - var s = window.getSelection(); - if (s.rangeCount) { - for (var i = 0; i < s.rangeCount; i++) { - s.removeRange(s.getRangeAt(i)); - } - } - - e.preventDefault(); - e.stopPropagation(); - } else { - // No floating action found - editorAction = false; - } - - // Verify current components tracking - 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; - } - - // Which component I am clicking - var path = e.path || (e.composedPath && e.composedPath()); - - // If path available get the first element in the chain - if (path) { - element = path[0]; - } else { - // Try to guess using the coordinates - if (e.target && e.target.shadowRoot) { - var d = e.target.shadowRoot; - } else { - var d = document; - } - // Get the first target element - element = d.elementFromPoint(x, y); - } - - isOpened(element); - } - - var mouseUp = function(e) { - if (editorAction && editorAction.e) { - if (typeof(editorAction.e.refresh) == 'function' && state.actioned) { - editorAction.e.refresh(); - } - editorAction.e.style.cursor = ''; - } - - // Reset - state = { - x: null, - y: null, - } - - editorAction = false; - } - - var mouseMove = function(e) { - if (editorAction) { - var x = e.clientX || e.pageX; - var y = e.clientY || e.pageY; - - // Action on going - if (! editorAction.resizing) { - if (state.x == null && state.y == null) { - state.x = x; - state.y = y; - } - - var dx = x - state.x; - var dy = y - state.y; - var top = editorAction.e.offsetTop + dy; - var left = editorAction.e.offsetLeft + dx; - - // Update position - editorAction.e.style.top = top + 'px'; - editorAction.e.style.left = left + 'px'; - editorAction.e.style.cursor = "move"; - - state.x = x; - state.y = y; - - - // Update element - if (typeof(editorAction.e.refresh) == 'function') { - state.actioned = true; - editorAction.e.refresh('position', top, left); - } - } else { - var width = null; - var height = null; - - if (editorAction.d == 'e-resize' || editorAction.d == 'ne-resize' || editorAction.d == 'se-resize') { - // Update width - width = editorAction.w + (x - editorAction.x); - editorAction.e.style.width = width + 'px'; - - // Update Height - if (e.shiftKey) { - var newHeight = (x - editorAction.x) * (editorAction.h / editorAction.w); - height = editorAction.h + newHeight; - editorAction.e.style.height = height + 'px'; - } else { - var newHeight = false; - } - } - - if (! newHeight) { - if (editorAction.d == 's-resize' || editorAction.d == 'se-resize' || editorAction.d == 'sw-resize') { - height = editorAction.h + (y - editorAction.y); - editorAction.e.style.height = height + 'px'; - } - } - - // Update element - if (typeof(editorAction.e.refresh) == 'function') { - state.actioned = true; - editorAction.e.refresh('dimensions', width, height); - } - } - } else { - // Resizing action - var item = jSuites.findElement(e.target, 'jpanel'); - // Found eligible component - if (item) { - if (item.getAttribute('tabindex')) { - var rect = item.getBoundingClientRect(); - if (e.clientY - rect.top < cornerSize) { - if (rect.width - (e.clientX - rect.left) < cornerSize) { - item.style.cursor = 'ne-resize'; - } else if (e.clientX - rect.left < cornerSize) { - item.style.cursor = 'nw-resize'; - } else { - item.style.cursor = 'n-resize'; - } - } else if (rect.height - (e.clientY - rect.top) < cornerSize) { - if (rect.width - (e.clientX - rect.left) < cornerSize) { - item.style.cursor = 'se-resize'; - } else if (e.clientX - rect.left < cornerSize) { - item.style.cursor = 'sw-resize'; - } else { - item.style.cursor = 's-resize'; - } - } else if (rect.width - (e.clientX - rect.left) < cornerSize) { - item.style.cursor = 'e-resize'; - } else if (e.clientX - rect.left < cornerSize) { - item.style.cursor = 'w-resize'; - } else { - item.style.cursor = ''; - } - } - } - } - } - - var mouseOver = function(e) { - var message = e.target.getAttribute('data-tooltip'); - if (message) { - // Instructions - tooltip.innerText = message; - - // Position - 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; - } - - tooltip.style.top = y + 'px'; - tooltip.style.left = x + 'px'; - document.body.appendChild(tooltip); - } else if (tooltip.innerText) { - tooltip.innerText = ''; - document.body.removeChild(tooltip); - } - } - - var dblClick = function(e) { - var item = jSuites.findElement(e.target, 'jpanel'); - if (item && typeof(item.dblclick) == 'function') { - // Create edition - item.dblclick(e); - } - } - - var contextMenu = function(e) { - var item = document.activeElement; - if (item && typeof(item.contextmenu) == 'function') { - // Create edition - item.contextmenu(e); - - e.preventDefault(); - e.stopImmediatePropagation(); - } else { - // Search for possible context menus - item = jSuites.findElement(e.target, function(o) { - return o.tagName && o.getAttribute('aria-contextmenu-id'); - }); - - if (item) { - var o = document.querySelector('#' + item); - if (! o) { - console.error('JSUITES: contextmenu id not found: ' + item); - } else { - o.contextmenu.open(e); - e.preventDefault(); - e.stopImmediatePropagation(); - } - } - } - } - - var keyDown = function(e) { - var item = document.activeElement; - if (item) { - if (e.key == "Delete" && typeof(item.delete) == 'function') { - item.delete(); - e.preventDefault(); - e.stopImmediatePropagation(); - } - } - - if (document.jsuitesComponents && document.jsuitesComponents.length) { - if (item = document.jsuitesComponents[document.jsuitesComponents.length - 1]) { - if (e.key == "Escape" && typeof(item.isOpened) == 'function' && typeof(item.close) == 'function') { - if (item.isOpened()) { - item.close(); - e.preventDefault(); - e.stopImmediatePropagation(); - } - } - } - } - } - - document.addEventListener('mouseup', mouseUp); - document.addEventListener("mousedown", mouseDown); - document.addEventListener('mousemove', mouseMove); - document.addEventListener('mouseover', mouseOver); - document.addEventListener('dblclick', dblClick); - document.addEventListener('keydown', keyDown); - document.addEventListener('contextmenu', contextMenu); -} - -/** - * Global jsuites event - */ -if (typeof(document) !== "undefined" && ! document.jsuitesComponents) { - Events(); -} - -jSuites.version = Version; - -jSuites.setExtensions = function(o) { - if (typeof(o) == 'object') { - var k = Object.keys(o); - for (var i = 0; i < k.length; i++) { - jSuites[k[i]] = o[k[i]]; - } - } -} - -jSuites.tracking = function(component, state) { - if (state == true) { - document.jsuitesComponents = document.jsuitesComponents.filter(function(v) { - return v !== null; - }); - - // Start after all events - setTimeout(function() { - document.jsuitesComponents.push(component); - }, 0); - - } else { - var index = document.jsuitesComponents.indexOf(component); - if (index >= 0) { - document.jsuitesComponents.splice(index, 1); - } - } -} - -/** - * Get or set a property from a JSON from a string. - */ -jSuites.path = function(str, val) { - str = str.split('.'); - if (str.length) { - var o = this; - var p = null; - while (str.length > 1) { - // Get the property - p = str.shift(); - // Check if the property exists - if (o.hasOwnProperty(p)) { - o = o[p]; - } else { - // Property does not exists - if (val === undefined) { - return undefined; - } else { - // Create the property - o[p] = {}; - // Next property - o = o[p]; - } - } - } - // Get the property - p = str.shift(); - // Set or get the value - if (val !== undefined) { - o[p] = val; - // Success - return true; - } else { - // Return the value - if (o) { - return o[p]; - } - } - } - // Something went wrong - return false; -} - -// Update dictionary -jSuites.setDictionary = function(d) { - if (! document.dictionary) { - document.dictionary = {} - } - // Replace the key into the dictionary and append the new ones - var k = Object.keys(d); - for (var i = 0; i < k.length; i++) { - document.dictionary[k[i]] = d[k[i]]; - } - - // Translations - var t = null; - for (var i = 0; i < jSuites.calendar.weekdays.length; i++) { - t = jSuites.translate(jSuites.calendar.weekdays[i]); - if (jSuites.calendar.weekdays[i]) { - jSuites.calendar.weekdays[i] = t; - jSuites.calendar.weekdaysShort[i] = t.substr(0,3); - } - } - for (var i = 0; i < jSuites.calendar.months.length; i++) { - t = jSuites.translate(jSuites.calendar.months[i]); - if (t) { - jSuites.calendar.months[i] = t; - jSuites.calendar.monthsShort[i] = t.substr(0,3); - } - } -} - -// Translate -jSuites.translate = function(t) { - if (typeof(document) !== "undefined" && document.dictionary) { - return document.dictionary[t] || t; - } else { - return t; - } -} - -jSuites.ajax = (function(options, complete) { - if (Array.isArray(options)) { - // Create multiple request controller - var multiple = { - instance: [], - complete: complete, - } - - if (options.length > 0) { - for (var i = 0; i < options.length; i++) { - options[i].multiple = multiple; - multiple.instance.push(jSuites.ajax(options[i])); - } - } - - return multiple; - } - - if (! options.data) { - options.data = {}; - } - - if (options.type) { - options.method = options.type; - } - - // Default method - if (! options.method) { - options.method = 'GET'; - } - - // Default type - if (! options.dataType) { - options.dataType = 'json'; - } - - if (options.data) { - // Parse object to variables format - var parseData = function (value, key) { - var vars = []; - if (value) { - var keys = Object.keys(value); - if (keys.length) { - for (var i = 0; i < keys.length; i++) { - if (key) { - var k = key + '[' + keys[i] + ']'; - } else { - var k = keys[i]; - } - - if (value[k] instanceof FileList) { - vars[k] = value[keys[i]]; - } else if (value[keys[i]] === null || value[keys[i]] === undefined) { - vars[k] = ''; - } else if (typeof(value[keys[i]]) == 'object') { - var r = parseData(value[keys[i]], k); - var o = Object.keys(r); - for (var j = 0; j < o.length; j++) { - vars[o[j]] = r[o[j]]; - } - } else { - vars[k] = value[keys[i]]; - } - } - } - } - - return vars; - } - - var d = parseData(options.data); - var k = Object.keys(d); - - // Data form - if (options.method == 'GET') { - if (k.length) { - var data = []; - for (var i = 0; i < k.length; i++) { - data.push(k[i] + '=' + encodeURIComponent(d[k[i]])); - } - - if (options.url.indexOf('?') < 0) { - options.url += '?'; - } - options.url += data.join('&'); - } - } else { - var data = new FormData(); - for (var i = 0; i < k.length; i++) { - if (d[k[i]] instanceof FileList) { - if (d[k[i]].length) { - for (var j = 0; j < d[k[i]].length; j++) { - data.append(k[i], d[k[i]][j], d[k[i]][j].name); - } - } - } else { - data.append(k[i], d[k[i]]); - } - } - } - } - - var httpRequest = new XMLHttpRequest(); - httpRequest.open(options.method, options.url, true); - httpRequest.setRequestHeader('X-Requested-With', 'XMLHttpRequest'); - - // Content type - if (options.contentType) { - httpRequest.setRequestHeader('Content-Type', options.contentType); - } - - // Headers - if (options.method == 'POST') { - httpRequest.setRequestHeader('Accept', 'application/json'); - } else { - if (options.dataType == 'blob') { - httpRequest.responseType = "blob"; - } else { - if (! options.contentType) { - if (options.dataType == 'json') { - httpRequest.setRequestHeader('Content-Type', 'text/json'); - } else if (options.dataType == 'html') { - httpRequest.setRequestHeader('Content-Type', 'text/html'); - } - } - } - } - - // 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); - } - - // Before send - if (typeof(jSuites.ajax.beforeSend) == 'function') { - jSuites.ajax.beforeSend(httpRequest); - } - - if (document.ajax && typeof(document.ajax.beforeSend) == 'function') { - document.ajax.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(err, result); - } - } - } else { - if (options.dataType == 'blob') { - var result = httpRequest.response; - } 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, httpRequest.status); - } - } - - // Global queue - if (jSuites.ajax.queue && jSuites.ajax.queue.length > 0) { - jSuites.ajax.send(jSuites.ajax.queue.shift()); - } - - // Global complete method - if (jSuites.ajax.requests && jSuites.ajax.requests.length) { - // Get index of this request in the container - var index = jSuites.ajax.requests.indexOf(httpRequest); - // Remove from the ajax requests container - jSuites.ajax.requests.splice(index, 1); - // Deprected: Last one? - if (! jSuites.ajax.requests.length) { - // Object event - if (options.complete && typeof(options.complete) == 'function') { - options.complete(result); - } - } - // Group requests - if (options.group) { - if (jSuites.ajax.oncomplete && typeof(jSuites.ajax.oncomplete[options.group]) == 'function') { - if (! jSuites.ajax.pending(options.group)) { - jSuites.ajax.oncomplete[options.group](); - jSuites.ajax.oncomplete[options.group] = null; - } - } - } - // Multiple requests controller - if (options.multiple && options.multiple.instance) { - // Get index of this request in the container - var index = options.multiple.instance.indexOf(httpRequest); - // Remove from the ajax requests container - options.multiple.instance.splice(index, 1); - // If this is the last one call method complete - if (! options.multiple.instance.length) { - if (options.multiple.complete && typeof(options.multiple.complete) == 'function') { - options.multiple.complete(result); - } - } - } - } - } - - // Keep the options - httpRequest.options = options; - // Data - httpRequest.data = data; - - // Queue - if (options.queue == true && jSuites.ajax.requests.length > 0) { - jSuites.ajax.queue.push(httpRequest); - } else { - jSuites.ajax.send(httpRequest) - } - - return httpRequest; -}); - -jSuites.ajax.send = function(httpRequest) { - if (httpRequest.data) { - if (Array.isArray(httpRequest.data)) { - httpRequest.send(httpRequest.data.join('&')); - } else { - httpRequest.send(httpRequest.data); - } - } else { - httpRequest.send(); - } - - jSuites.ajax.requests.push(httpRequest); -} - -jSuites.ajax.exists = function(url, __callback) { - var http = new XMLHttpRequest(); - http.open('HEAD', url, false); - http.send(); - if (http.status) { - __callback(http.status); - } -} - -jSuites.ajax.pending = function(group) { - var n = 0; - var o = jSuites.ajax.requests; - if (o && o.length) { - for (var i = 0; i < o.length; i++) { - if (! group || group == o[i].options.group) { - n++ - } - } - } - return n; -} - -jSuites.ajax.oncomplete = {}; -jSuites.ajax.requests = []; -jSuites.ajax.queue = []; - -jSuites.alert = function(message) { - if (jSuites.getWindowWidth() < 800 && jSuites.dialog) { - jSuites.dialog.open({ - title:'Alert', - message:message, - }); - } else { - alert(message); - } -} - -jSuites.animation = {}; - -jSuites.animation.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); - } -} - -jSuites.animation.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); - } -} - -jSuites.animation.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); - } -} - -jSuites.animation.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); - } -} - -jSuites.animation.fadeIn = function(element, done) { - element.style.display = ''; - element.classList.add('fade-in'); - setTimeout(function() { - element.classList.remove('fade-in'); - if (typeof(done) == 'function') { - done(); - } - }, 2000); -} - -jSuites.animation.fadeOut = function(element, done) { - element.classList.add('fade-out'); - setTimeout(function() { - element.style.display = 'none'; - element.classList.remove('fade-out'); - if (typeof(done) == 'function') { - done(); - } - }, 1000); -} - -jSuites.calendar = (function(el, options) { - // Already created, update options - if (el.calendar) { - return el.calendar.setOptions(options, true); - } - - // New instance - var obj = { type:'calendar' }; - obj.options = {}; - - // Date - obj.date = null; - - /** - * Update options - */ - obj.setOptions = function(options, reset) { - // Default configuration - var defaults = { - // Render type: [ default | year-month-picker ] - type: 'default', - // 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: jSuites.calendar.monthsShort, - monthsFull: jSuites.calendar.months, - weekdays: jSuites.calendar.weekdays, - textDone: jSuites.translate('Done'), - textReset: jSuites.translate('Reset'), - textUpdate: jSuites.translate('Update'), - // Value - value: null, - // Fullscreen (this is automatic set for screensize < 800) - fullscreen: false, - // Create the calendar closed as default - opened: false, - // Events - onopen: null, - onclose: null, - onchange: null, - onupdate: null, - // Internal mode controller - mode: null, - position: null, - // Data type - dataType: null, - // Controls - controls: true, - } - - // Loop through our object - for (var property in defaults) { - if (options && options.hasOwnProperty(property)) { - obj.options[property] = options[property]; - } else { - if (typeof(obj.options[property]) == 'undefined' || reset === true) { - obj.options[property] = defaults[property]; - } - } - } - - // Reset button - if (obj.options.resetButton == false) { - calendarReset.style.display = 'none'; - } else { - calendarReset.style.display = ''; - } - - // Readonly - if (obj.options.readonly) { - el.setAttribute('readonly', 'readonly'); - } else { - el.removeAttribute('readonly'); - } - - // Placeholder - if (obj.options.placeholder) { - el.setAttribute('placeholder', obj.options.placeholder); - } else { - el.removeAttribute('placeholder'); - } - - if (jSuites.isNumeric(obj.options.value) && obj.options.value > 0) { - obj.options.value = jSuites.calendar.numToDate(obj.options.value); - // Data type numeric - obj.options.dataType = 'numeric'; - } - - // Texts - calendarReset.innerHTML = obj.options.textReset; - calendarConfirm.innerHTML = obj.options.textDone; - calendarControlsUpdateButton.innerHTML = obj.options.textUpdate; - - // Define mask - el.setAttribute('data-mask', obj.options.format.toLowerCase()); - - // Value - if (! obj.options.value && obj.options.today) { - var value = jSuites.calendar.now(); - } else { - var value = obj.options.value; - } - - // Set internal date - if (value) { - // Force the update - obj.options.value = null; - // New value - obj.setValue(value); - } - - return obj; - } - - /** - * Open the calendar - */ - obj.open = function (value) { - if (! calendar.classList.contains('jcalendar-focus')) { - if (! calendar.classList.contains('jcalendar-inline')) { - // Current - jSuites.calendar.current = obj; - // Start tracking - jSuites.tracking(obj, true); - // Create the days - obj.getDays(); - // Render months - if (obj.options.type == 'year-month-picker') { - obj.getMonths(); - } - // Get time - if (obj.options.time) { - calendarSelectHour.value = obj.date[3]; - calendarSelectMin.value = obj.date[4]; - } - - // Show calendar - calendar.classList.add('jcalendar-focus'); - - // Get the position of the corner helper - if (jSuites.getWindowWidth() < 800 || obj.options.fullscreen) { - calendar.classList.add('jcalendar-fullsize'); - // Animation - jSuites.animation.slideBottom(calendarContent, 1); - } else { - calendar.classList.remove('jcalendar-fullsize'); - - var rect = el.getBoundingClientRect(); - var 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) { - var d = -1 * (rect.height + rectContent.height + 2); - if (d + rect.top < 0) { - d = -1 * (rect.top + rect.height); - } - calendarContainer.style.top = d + 'px'; - } else { - calendarContainer.style.top = 2 + 'px'; - } - - if (window.innerWidth < rect.left + rectContent.width) { - var d = window.innerWidth - (rect.left + rectContent.width + 20); - calendarContainer.style.left = d + 'px'; - } else { - calendarContainer.style.left = '0px'; - } - } - } - - // Events - if (typeof(obj.options.onopen) == 'function') { - obj.options.onopen(el); - } - } - } - } - - obj.close = function (ignoreEvents, update) { - if (calendar.classList.contains('jcalendar-focus')) { - 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'); - // Stop tracking - jSuites.tracking(obj, false); - // Current - jSuites.calendar.current = null; - } - - 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 if (obj.options.mode == 'months') { - obj.date[0] = parseInt(obj.date[0]) - 1; - // Update picker table of months - obj.getMonths(); - } 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 if (obj.options.mode == 'months') { - obj.date[0] = parseInt(obj.date[0]) + 1; - // Update picker table of months - obj.getMonths(); - } 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(); - } - } - - /** - * Set today - */ - obj.setToday = function() { - // Today - var value = new Date().toISOString().substr(0, 10); - // Change value - obj.setValue(value); - // Value - return value; - } - - obj.setValue = function(val) { - if (! val) { - val = '' + val; - } - // Values - var newValue = val; - var oldValue = obj.options.value; - - if (oldValue != newValue) { - // Set label - if (! newValue) { - obj.date = null; - var val = ''; - el.classList.remove('jcalendar_warning'); - el.title = ''; - } else { - var value = obj.setLabel(newValue, obj.options); - 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); - - // Current selection day - var current = jSuites.calendar.now(new Date(y, m-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)) { - el.classList.add('jcalendar_warning'); - el.title = jSuites.translate('Date outside the valid range'); - } else { - el.classList.remove('jcalendar_warning'); - el.title = ''; - } - } else { - el.classList.remove('jcalendar_warning'); - el.title = ''; - } - } - - // New value - obj.options.value = newValue; - - if (typeof(obj.options.onchange) == 'function') { - obj.options.onchange(el, newValue, oldValue); - } - - // Lemonade JS - if (el.value != val) { - el.value = val; - if (typeof(el.oninput) == 'function') { - el.oninput({ - type: 'input', - target: el, - value: el.value - }); - } - } - } - - obj.getDays(); - // Render months - if (obj.options.type == 'year-month-picker') { - obj.getMonths(); - } - } - - obj.getValue = function() { - if (obj.date) { - if (obj.options.time) { - return jSuites.two(obj.date[0]) + '-' + jSuites.two(obj.date[1]) + '-' + jSuites.two(obj.date[2]) + ' ' + jSuites.two(obj.date[3]) + ':' + jSuites.two(obj.date[4]) + ':' + jSuites.two(0); - } else { - return jSuites.two(obj.date[0]) + '-' + jSuites.two(obj.date[1]) + '-' + jSuites.two(obj.date[2]) + ' ' + jSuites.two(0) + ':' + jSuites.two(0) + ':' + jSuites.two(0); - } - } else { - return ""; - } - } - - /** - * Calendar - */ - obj.update = function(element, v) { - if (element.classList.contains('jcalendar-disabled')) { - // Do nothing - } else { - var elements = calendar.querySelector('.jcalendar-selected'); - if (elements) { - elements.classList.remove('jcalendar-selected'); - } - element.classList.add('jcalendar-selected'); - - if (element.classList.contains('jcalendar-set-month')) { - obj.date[1] = v; - obj.date[2] = 1; // first day of the month - } else { - obj.date[2] = element.innerText; - } - - if (! obj.options.time) { - obj.close(); - } else { - obj.date[3] = calendarSelectHour.value; - obj.date[4] = calendarSelectMin.value; - } - } - - // Update - updateActions(); - } - - /** - * Set to blank - */ - obj.reset = function() { - // Close calendar - obj.setValue(''); - obj.date = null; - obj.close(false, 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 && jSuites.isNumeric(obj.date[0]) ? obj.date[0] : parseInt(date.getFullYear()); - var month = obj.date && jSuites.isNumeric(obj.date[1]) ? obj.date[1] : parseInt(date.getMonth()) + 1; - var day = obj.date && jSuites.isNumeric(obj.date[2]) ? obj.date[2] : parseInt(date.getDate()); - var hour = obj.date && jSuites.isNumeric(obj.date[3]) ? obj.date[3] : parseInt(date.getHours()); - var min = obj.date && jSuites.isNumeric(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[index].substr(0,1); - 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 < 6; j++) { - // Reset cells container - var row = document.createElement('tr'); - row.setAttribute('align', 'center'); - row.style.height = '34px'; - - // 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'); - } - } - } - // Day cell - row.appendChild(cell); - // Index - index++; - } - - // Add cell to the calendar body - 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; - - // Value - var value = obj.options.value; - - // Current date - var date = new Date(); - var currentYear = parseInt(date.getFullYear()); - var currentMonth = parseInt(date.getMonth()) + 1; - var selectedYear = obj.date && jSuites.isNumeric(obj.date[0]) ? obj.date[0] : currentYear; - var selectedMonth = obj.date && jSuites.isNumeric(obj.date[1]) ? obj.date[1] : currentMonth; - - // Update title - calendarLabelYear.innerHTML = obj.date[0]; - calendarLabelMonth.innerHTML = months[selectedMonth-1]; - - // Table - var table = document.createElement('table'); - table.setAttribute('width', '100%'); - - // Row - var row = null; - - // Calendar table - for (var i = 0; i < 12; i++) { - if (! (i % 4)) { - // Reset cells container - var row = document.createElement('tr'); - row.setAttribute('align', 'center'); - table.appendChild(row); - } - - // Create cell - var cell = document.createElement('td'); - cell.classList.add('jcalendar-set-month'); - cell.setAttribute('data-value', i+1); - cell.innerText = months[i]; - - if (obj.options.validRange) { - var current = selectedYear + '-' + jSuites.two(i+1); - if (! obj.options.validRange[0] || current >= obj.options.validRange[0].substr(0,7)) { - var test1 = true; - } else { - var test1 = false; - } - - if (! obj.options.validRange[1] || current <= obj.options.validRange[1].substr(0,7)) { - var test2 = true; - } else { - var test2 = false; - } - - if (! (test1 && test2)) { - cell.classList.add('jcalendar-disabled'); - } - } - - if (i+1 == selectedMonth) { - cell.classList.add('jcalendar-selected'); - } - - if (currentYear == selectedYear && i+1 == currentMonth) { - cell.style.fontWeight = 'bold'; - } - - row.appendChild(cell); - } - - calendarBody.innerHTML = '