├── .editorconfig ├── .eslintignore ├── .eslintrc ├── .github └── release.yml ├── .gitignore ├── .npmrc ├── Book1.xlsx ├── LICENSE ├── README.md ├── SampleData.xlsx ├── docs └── screen01.png ├── esbuild.config.mjs ├── main2.css ├── manifest-beta.json ├── manifest.json ├── officeTheme.xml ├── package-lock.json ├── package.json ├── spreadsheetData.json ├── src ├── Globals.d.ts ├── SettingTab.ts ├── Settings.ts ├── Views │ ├── SheetView.ts │ ├── save.svg │ └── spreadSheetWrapper.ts ├── main.ts └── utils │ ├── excelColors.ts │ ├── excelConverter.ts │ └── xlsxpread.js ├── styles.scss ├── tsconfig.json ├── version-bump.mjs ├── versions.json └── wb.json /.editorconfig: -------------------------------------------------------------------------------- 1 | # top-most EditorConfig file 2 | root = true 3 | 4 | [*] 5 | charset = utf-8 6 | end_of_line = lf 7 | insert_final_newline = true 8 | indent_style = space 9 | indent_size = 4 10 | tab_width = 4 11 | -------------------------------------------------------------------------------- /.eslintignore: -------------------------------------------------------------------------------- 1 | npm node_modules 2 | build -------------------------------------------------------------------------------- /.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "root": true, 3 | "parser": "@typescript-eslint/parser", 4 | "env": { "node": true }, 5 | "plugins": [ 6 | "@typescript-eslint" 7 | ], 8 | "extends": [ 9 | "eslint:recommended", 10 | "plugin:@typescript-eslint/eslint-recommended", 11 | "plugin:@typescript-eslint/recommended" 12 | ], 13 | "parserOptions": { 14 | "sourceType": "module" 15 | }, 16 | "rules": { 17 | "no-unused-vars": "off", 18 | "@typescript-eslint/no-unused-vars": ["error", { "args": "none" }], 19 | "@typescript-eslint/ban-ts-comment": "off", 20 | "no-prototype-builtins": "off", 21 | "@typescript-eslint/no-empty-function": "off" 22 | } 23 | } -------------------------------------------------------------------------------- /.github/release.yml: -------------------------------------------------------------------------------- 1 | name: Release Obsidian plugin 2 | 3 | on: 4 | push: 5 | tags: 6 | - "*" 7 | 8 | jobs: 9 | build: 10 | runs-on: ubuntu-latest 11 | 12 | steps: 13 | - uses: actions/checkout@v3 14 | 15 | - name: Use Node.js 16 | uses: actions/setup-node@v3 17 | with: 18 | node-version: "18.x" 19 | 20 | - name: Build plugin 21 | run: | 22 | npm install 23 | npm run build 24 | 25 | - name: Create release 26 | env: 27 | GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} 28 | run: | 29 | tag="${GITHUB_REF#refs/tags/}" 30 | 31 | gh release create "$tag" \ 32 | --title="$tag" \ 33 | --draft \ 34 | main.js manifest.json styles.css 35 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # vscode 2 | .vscode 3 | 4 | # Intellij 5 | *.iml 6 | .idea 7 | 8 | # npm 9 | node_modules 10 | 11 | # Don't include the compiled main.js file in the repo. 12 | # They should be uploaded to GitHub releases instead. 13 | main.js 14 | styles.css 15 | 16 | # Exclude sourcemaps 17 | *.map 18 | 19 | # obsidian 20 | data.json 21 | 22 | # Exclude macOS Finder (System Explorer) View States 23 | .DS_Store 24 | -------------------------------------------------------------------------------- /.npmrc: -------------------------------------------------------------------------------- 1 | tag-version-prefix="" -------------------------------------------------------------------------------- /Book1.xlsx: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Canna71/obsidian-sheets/4f33d6c741fb0ab0dfbd877586dc525c7c3c8518/Book1.xlsx -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) [2023] [Gabriele Cannata] 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Obsidian Sheets 2 | 3 | ![](docs/screen01.png) 4 | 5 | This plugin enables to work with tabular data directly in Obsidian: either storing data in the note itself, or on an external Excel or CSV file. 6 | The following files are supported: 7 | - XLSX 8 | - XLS 9 | - CSV 10 | 11 | 12 | ## Code Block Parameters 13 | 14 | In order to create a sheet it is enough to create a code block with the `sheet` name 15 | 16 | ~~~markdown 17 | ```sheet 18 | 19 | ``` 20 | ~~~ 21 | 22 | Following is the complete list of properties with their default values: 23 | 24 | ~~~YAML 25 | ```sheet 26 | filename: 27 | enableSave: 28 | autoSave: 29 | height: 540 30 | width: "auto" 31 | rows: 100 32 | cols: 26 33 | fontSize: 10 34 | cellHeight: 25 35 | cellWidth: 100 36 | ``` 37 | ~~~ 38 | 39 | 40 | If `filename` is provided, data is loaded from that file, if already existing. Otherwise it will be created upon save. 41 | If `filename` is not provided, data will be stored in the code block itself. 42 | 43 | Note that by default saving data to external files is disabled because - in general - there will be loss of information. This could be enabled at vault-level via the settings, or code-block level by specifying `enableSave: true` in the code block. 44 | When saving to file, by default it will also autosave after each modification. When saving to the code block in order to save one must click the save icon in the top left. 45 | 46 | ## Supported Formats 47 | 48 | When saving to code block, all formatting and changes done to the sheet are perserved. 49 | When saving to `XLSX` most formatting and data will be saved and read back. Note however that when reading an existing `XLSX` file, only a subset of functionality will be supported. 50 | Support of `XLS` is limited to data and cell merges 51 | CSV can, of course, only store data. 52 | 53 | 54 | ## Attributions 55 | 56 | This plugin uses the following libraries: 57 | 58 | https://github.com/SheetJS 59 | 60 | https://github.com/myliang/x-spreadsheet (forked here: https://github.com/Canna71/x-spreadsheet) 61 | 62 | https://github.com/exceljs/exceljs 63 | -------------------------------------------------------------------------------- /SampleData.xlsx: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Canna71/obsidian-sheets/4f33d6c741fb0ab0dfbd877586dc525c7c3c8518/SampleData.xlsx -------------------------------------------------------------------------------- /docs/screen01.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Canna71/obsidian-sheets/4f33d6c741fb0ab0dfbd877586dc525c7c3c8518/docs/screen01.png -------------------------------------------------------------------------------- /esbuild.config.mjs: -------------------------------------------------------------------------------- 1 | import esbuild from "esbuild"; 2 | import process from "process"; 3 | import builtins from 'builtin-modules' 4 | import {sassPlugin} from 'esbuild-sass-plugin' 5 | // import svgrPlugin from 'esbuild-plugin-svgr'; 6 | import { lessLoader } from 'esbuild-plugin-less'; 7 | // eslint-disable-next-line @typescript-eslint/no-var-requires 8 | // import * as path from 'path' 9 | import alias from 'esbuild-plugin-alias'; 10 | import path from 'path'; 11 | import { fileURLToPath } from 'url'; 12 | 13 | const __filename = fileURLToPath(import.meta.url); 14 | const __dirname = path.dirname(__filename); 15 | 16 | const banner = 17 | `/* 18 | THIS IS A GENERATED/BUNDLED FILE BY ESBUILD 19 | if you want to view the source, please visit the github repository of this plugin 20 | */ 21 | `; 22 | 23 | const prod = (process.argv[2] === 'production'); 24 | 25 | esbuild.build({ 26 | banner: { 27 | js: banner, 28 | }, 29 | entryPoints: ['src/main.ts'], 30 | bundle: true, 31 | minify: prod, 32 | external: [ 33 | 'obsidian', 34 | 'electron', 35 | '@codemirror/autocomplete', 36 | '@codemirror/collab', 37 | '@codemirror/commands', 38 | '@codemirror/language', 39 | '@codemirror/lint', 40 | '@codemirror/search', 41 | '@codemirror/state', 42 | '@codemirror/view', 43 | '@lezer/common', 44 | '@lezer/highlight', 45 | '@lezer/lr', 46 | ...builtins], 47 | format: 'cjs', 48 | watch: !prod, 49 | target: 'es2016', 50 | logLevel: "info", 51 | sourcemap: prod ? false : 'inline', 52 | treeShaking: true, 53 | outfile: 'main.js', 54 | // https://github.com/glromeo/esbuild-sass-plugin#--rewriting-relative-urls 55 | plugins: [ 56 | lessLoader({}), 57 | alias({ 58 | 'x-data-spreadsheet': path.resolve(__dirname,'node_modules/x-data-spreadsheet/dist/xspreadsheet.js') 59 | }), 60 | ], 61 | loader: { 62 | '.ts': 'ts', 63 | '.svg': 'text', 64 | } 65 | }).catch(() => process.exit(1)); 66 | 67 | esbuild.build({ 68 | entryPoints: ['styles.scss'], 69 | outfile: "styles.css", 70 | // outdir: "/", 71 | watch: !prod, 72 | plugins: [sassPlugin()] 73 | }).catch(() => process.exit(1)); 74 | // node_modules/x-data-spreadsheet/src/index.less 75 | 76 | /* 77 | 78 | { 79 | precompile(source, pathname) { 80 | const basedir = path.dirname(pathname) 81 | return source.replace(/(url\(['"]?)(\.\.?\/)([^'")]+['"]?\))/g, `$1${basedir}/$2$3`) 82 | } 83 | } 84 | */ 85 | -------------------------------------------------------------------------------- /main2.css: -------------------------------------------------------------------------------- 1 | /* node_modules/x-data-spreadsheet/src/index.less */ 2 | body { 3 | margin: 0; 4 | } 5 | .x-spreadsheet { 6 | font-size: 13px; 7 | line-height: normal; 8 | user-select: none; 9 | -moz-user-select: none; 10 | font-family: 11 | "Lato", 12 | "Source Sans Pro", 13 | Roboto, 14 | Helvetica, 15 | Arial, 16 | sans-serif; 17 | box-sizing: content-box; 18 | background: #fff; 19 | -webkit-font-smoothing: antialiased; 20 | } 21 | .x-spreadsheet textarea { 22 | font: 23 | 400 13px Arial, 24 | "Lato", 25 | "Source Sans Pro", 26 | Roboto, 27 | Helvetica, 28 | sans-serif; 29 | } 30 | .x-spreadsheet-sheet { 31 | position: relative; 32 | overflow: hidden; 33 | } 34 | .x-spreadsheet-table { 35 | vertical-align: bottom; 36 | } 37 | .x-spreadsheet-tooltip { 38 | font-family: inherit; 39 | position: absolute; 40 | padding: 5px 10px; 41 | color: #fff; 42 | border-radius: 1px; 43 | background: #000000; 44 | font-size: 12px; 45 | z-index: 201; 46 | } 47 | .x-spreadsheet-tooltip:before { 48 | pointer-events: none; 49 | position: absolute; 50 | left: calc(50% - 4px); 51 | top: -4px; 52 | content: ""; 53 | width: 8px; 54 | height: 8px; 55 | background: inherit; 56 | -webkit-transform: rotate(45deg); 57 | transform: rotate(45deg); 58 | z-index: 1; 59 | box-shadow: 1px 1px 3px -1px rgba(0, 0, 0, 0.3); 60 | } 61 | .x-spreadsheet-color-palette { 62 | padding: 5px; 63 | } 64 | .x-spreadsheet-color-palette table { 65 | margin: 0; 66 | padding: 0; 67 | border-collapse: separate; 68 | border-spacing: 2; 69 | background: #fff; 70 | } 71 | .x-spreadsheet-color-palette table td { 72 | margin: 0; 73 | cursor: pointer; 74 | border: 1px solid transparent; 75 | } 76 | .x-spreadsheet-color-palette table td:hover { 77 | border-color: #ddd; 78 | } 79 | .x-spreadsheet-color-palette table td .x-spreadsheet-color-palette-cell { 80 | width: 16px; 81 | height: 16px; 82 | } 83 | .x-spreadsheet-border-palette { 84 | padding: 6px; 85 | } 86 | .x-spreadsheet-border-palette table { 87 | margin: 0; 88 | padding: 0; 89 | border-collapse: separate; 90 | border-spacing: 0; 91 | background: #fff; 92 | table-layout: fixed; 93 | } 94 | .x-spreadsheet-border-palette table td { 95 | margin: 0; 96 | } 97 | .x-spreadsheet-border-palette .x-spreadsheet-border-palette-left { 98 | border-right: 1px solid #eee; 99 | padding-right: 6px; 100 | } 101 | .x-spreadsheet-border-palette .x-spreadsheet-border-palette-left .x-spreadsheet-border-palette-cell { 102 | width: 30px; 103 | height: 30px; 104 | cursor: pointer; 105 | text-align: center; 106 | } 107 | .x-spreadsheet-border-palette .x-spreadsheet-border-palette-left .x-spreadsheet-border-palette-cell .x-spreadsheet-icon-img { 108 | opacity: 0.8; 109 | } 110 | .x-spreadsheet-border-palette .x-spreadsheet-border-palette-left .x-spreadsheet-border-palette-cell:hover { 111 | background-color: #eee; 112 | } 113 | .x-spreadsheet-border-palette .x-spreadsheet-border-palette-right { 114 | padding-left: 6px; 115 | } 116 | .x-spreadsheet-border-palette .x-spreadsheet-border-palette-right .x-spreadsheet-toolbar-btn { 117 | margin-top: 0; 118 | margin-bottom: 3px; 119 | } 120 | .x-spreadsheet-border-palette .x-spreadsheet-border-palette-right .x-spreadsheet-line-type { 121 | position: relative; 122 | left: 0; 123 | top: -3px; 124 | } 125 | .x-spreadsheet-dropdown { 126 | position: relative; 127 | } 128 | .x-spreadsheet-dropdown .x-spreadsheet-dropdown-content { 129 | position: absolute; 130 | z-index: 200; 131 | background: #fff; 132 | box-shadow: 1px 2px 5px 2px rgba(51, 51, 51, 0.15); 133 | } 134 | .x-spreadsheet-dropdown.bottom-left .x-spreadsheet-dropdown-content { 135 | top: calc(100% + 5px); 136 | left: 0; 137 | } 138 | .x-spreadsheet-dropdown.bottom-right .x-spreadsheet-dropdown-content { 139 | top: calc(100% + 5px); 140 | right: 0; 141 | } 142 | .x-spreadsheet-dropdown.top-left .x-spreadsheet-dropdown-content { 143 | bottom: calc(100% + 5px); 144 | left: 0; 145 | } 146 | .x-spreadsheet-dropdown.top-right .x-spreadsheet-dropdown-content { 147 | bottom: calc(100% + 5px); 148 | right: 0; 149 | } 150 | .x-spreadsheet-dropdown .x-spreadsheet-dropdown-title { 151 | padding: 0 5px; 152 | display: inline-block; 153 | } 154 | .x-spreadsheet-dropdown .x-spreadsheet-dropdown-header .x-spreadsheet-icon.arrow-left { 155 | margin-left: 4px; 156 | } 157 | .x-spreadsheet-dropdown .x-spreadsheet-dropdown-header .x-spreadsheet-icon.arrow-right { 158 | width: 10px; 159 | margin-right: 4px; 160 | } 161 | .x-spreadsheet-dropdown .x-spreadsheet-dropdown-header .x-spreadsheet-icon.arrow-right .arrow-down { 162 | left: -130px; 163 | } 164 | .x-spreadsheet-resizer { 165 | position: absolute; 166 | z-index: 11; 167 | } 168 | .x-spreadsheet-resizer .x-spreadsheet-resizer-hover { 169 | background-color: rgba(75, 137, 255, 0.25); 170 | } 171 | .x-spreadsheet-resizer .x-spreadsheet-resizer-line { 172 | position: absolute; 173 | } 174 | .x-spreadsheet-resizer.horizontal { 175 | cursor: row-resize; 176 | } 177 | .x-spreadsheet-resizer.horizontal .x-spreadsheet-resizer-line { 178 | border-bottom: 2px dashed #4b89ff; 179 | left: 0; 180 | bottom: 0; 181 | } 182 | .x-spreadsheet-resizer.vertical { 183 | cursor: col-resize; 184 | } 185 | .x-spreadsheet-resizer.vertical .x-spreadsheet-resizer-line { 186 | border-right: 2px dashed #4b89ff; 187 | top: 0; 188 | right: 0; 189 | } 190 | .x-spreadsheet-scrollbar { 191 | position: absolute; 192 | bottom: 0; 193 | right: 0; 194 | background-color: #f4f5f8; 195 | opacity: 0.9; 196 | z-index: 12; 197 | } 198 | .x-spreadsheet-scrollbar.horizontal { 199 | right: 15px; 200 | overflow-x: scroll; 201 | overflow-y: hidden; 202 | } 203 | .x-spreadsheet-scrollbar.horizontal > div { 204 | height: 1px; 205 | background: #ddd; 206 | } 207 | .x-spreadsheet-scrollbar.vertical { 208 | bottom: 15px; 209 | overflow-x: hidden; 210 | overflow-y: scroll; 211 | } 212 | .x-spreadsheet-scrollbar.vertical > div { 213 | width: 1px; 214 | background: #ddd; 215 | } 216 | .x-spreadsheet-overlayer { 217 | position: absolute; 218 | left: 0; 219 | top: 0; 220 | z-index: 10; 221 | } 222 | .x-spreadsheet-overlayer .x-spreadsheet-overlayer-content { 223 | position: absolute; 224 | overflow: hidden; 225 | pointer-events: none; 226 | width: 100%; 227 | height: 100%; 228 | } 229 | .x-spreadsheet-editor, 230 | .x-spreadsheet-selector { 231 | box-sizing: content-box; 232 | position: absolute; 233 | overflow: hidden; 234 | pointer-events: none; 235 | top: 0; 236 | left: 0; 237 | width: 100%; 238 | height: 100%; 239 | } 240 | .x-spreadsheet-selector .hide-input { 241 | position: absolute; 242 | z-index: 0; 243 | } 244 | .x-spreadsheet-selector .hide-input input { 245 | padding: 0; 246 | width: 0; 247 | border: none !important; 248 | } 249 | .x-spreadsheet-selector .x-spreadsheet-selector-area { 250 | position: absolute; 251 | border: 2px solid #4b89ff; 252 | background: rgba(75, 137, 255, 0.1); 253 | z-index: 5; 254 | } 255 | .x-spreadsheet-selector .x-spreadsheet-selector-clipboard, 256 | .x-spreadsheet-selector .x-spreadsheet-selector-autofill { 257 | position: absolute; 258 | background: transparent; 259 | z-index: 100; 260 | } 261 | .x-spreadsheet-selector .x-spreadsheet-selector-clipboard { 262 | border: 2px dashed #4b89ff; 263 | } 264 | .x-spreadsheet-selector .x-spreadsheet-selector-autofill { 265 | border: 1px dashed rgba(0, 0, 0, 0.45); 266 | } 267 | .x-spreadsheet-selector .x-spreadsheet-selector-corner { 268 | pointer-events: auto; 269 | position: absolute; 270 | cursor: crosshair; 271 | font-size: 0; 272 | height: 5px; 273 | width: 5px; 274 | right: -5px; 275 | bottom: -5px; 276 | border: 2px solid #ffffff; 277 | background: #4b89ff; 278 | } 279 | .x-spreadsheet-editor { 280 | z-index: 20; 281 | } 282 | .x-spreadsheet-editor .x-spreadsheet-editor-area { 283 | position: absolute; 284 | text-align: left; 285 | border: 2px solid #4b89ff; 286 | line-height: 0; 287 | z-index: 100; 288 | pointer-events: auto; 289 | } 290 | .x-spreadsheet-editor .x-spreadsheet-editor-area textarea { 291 | box-sizing: content-box; 292 | border: none; 293 | padding: 0 3px; 294 | outline: none; 295 | resize: none; 296 | text-align: start; 297 | overflow-y: hidden; 298 | font: 299 | 400 13px Arial, 300 | "Lato", 301 | "Source Sans Pro", 302 | Roboto, 303 | Helvetica, 304 | sans-serif; 305 | color: inherit; 306 | white-space: normal; 307 | word-wrap: break-word; 308 | line-height: 22px; 309 | margin: 0; 310 | } 311 | .x-spreadsheet-editor .x-spreadsheet-editor-area .textline { 312 | overflow: hidden; 313 | visibility: hidden; 314 | position: fixed; 315 | top: 0; 316 | left: 0; 317 | } 318 | .x-spreadsheet-item { 319 | user-select: none; 320 | background: 0; 321 | border: 1px solid transparent; 322 | outline: none; 323 | height: 26px; 324 | color: rgba(0, 0, 0, 0.9); 325 | line-height: 26px; 326 | list-style: none; 327 | padding: 2px 10px; 328 | cursor: default; 329 | text-align: left; 330 | overflow: hidden; 331 | } 332 | .x-spreadsheet-item.disabled { 333 | pointer-events: none; 334 | opacity: 0.5; 335 | } 336 | .x-spreadsheet-item:hover, 337 | .x-spreadsheet-item.active { 338 | background: rgba(0, 0, 0, 0.05); 339 | } 340 | .x-spreadsheet-item.divider { 341 | height: 0; 342 | padding: 0; 343 | margin: 5px 0; 344 | border: none; 345 | border-bottom: 1px solid rgba(0, 0, 0, 0.1); 346 | } 347 | .x-spreadsheet-item .label { 348 | float: right; 349 | opacity: 0.65; 350 | font-size: 1em; 351 | } 352 | .x-spreadsheet-item.state, 353 | .x-spreadsheet-header.state { 354 | padding-left: 35px !important; 355 | position: relative; 356 | } 357 | .x-spreadsheet-item.state:before, 358 | .x-spreadsheet-header.state:before { 359 | content: ""; 360 | position: absolute; 361 | width: 10px; 362 | height: 10px; 363 | left: 12px; 364 | top: calc(50% - 5px); 365 | background: rgba(0, 0, 0, 0.08); 366 | border-radius: 2px; 367 | } 368 | .x-spreadsheet-item.state.checked:before, 369 | .x-spreadsheet-header.state.checked:before { 370 | background: #4b89ff; 371 | } 372 | .x-spreadsheet-checkbox { 373 | position: relative; 374 | display: inline-block; 375 | backface-visibility: hidden; 376 | outline: 0; 377 | vertical-align: baseline; 378 | font-style: normal; 379 | font-size: 1rem; 380 | line-height: 1em; 381 | } 382 | .x-spreadsheet-checkbox > input { 383 | position: absolute; 384 | top: 0; 385 | left: 0; 386 | opacity: 0 !important; 387 | outline: 0; 388 | z-index: -1; 389 | } 390 | .x-spreadsheet-suggest, 391 | .x-spreadsheet-contextmenu, 392 | .x-spreadsheet-sort-filter { 393 | position: absolute; 394 | box-shadow: 1px 2px 5px 2px rgba(51, 51, 51, 0.15); 395 | background: #fff; 396 | z-index: 100; 397 | width: 260px; 398 | pointer-events: auto; 399 | overflow: auto; 400 | } 401 | .x-spreadsheet-suggest { 402 | width: 200px; 403 | } 404 | .x-spreadsheet-filter { 405 | border: 1px solid #e9e9e9; 406 | font-size: 12px; 407 | margin: 10px; 408 | } 409 | .x-spreadsheet-filter .x-spreadsheet-header { 410 | padding: 0.5em 0.75em; 411 | background: #f8f8f9; 412 | border-bottom: 1px solid #e9e9e9; 413 | border-left: 1px solid transparent; 414 | } 415 | .x-spreadsheet-filter .x-spreadsheet-body { 416 | height: 200px; 417 | overflow-y: auto; 418 | } 419 | .x-spreadsheet-filter .x-spreadsheet-body .x-spreadsheet-item { 420 | height: 20px; 421 | line-height: 20px; 422 | } 423 | .x-spreadsheet-sort-filter .x-spreadsheet-buttons { 424 | margin: 10px; 425 | } 426 | .x-spreadsheet-toolbar, 427 | .x-spreadsheet-bottombar { 428 | height: 40px; 429 | padding: 0 30px; 430 | text-align: left; 431 | background: #f5f6f7; 432 | display: flex; 433 | } 434 | .x-spreadsheet-bottombar { 435 | position: relative; 436 | border-top: 1px solid #e0e2e4; 437 | } 438 | .x-spreadsheet-bottombar .x-spreadsheet-menu > li { 439 | line-height: 40px; 440 | height: 40px; 441 | padding-top: 0; 442 | padding-bottom: 0; 443 | vertical-align: middle; 444 | border-right: 1px solid #e8eaed; 445 | } 446 | .x-spreadsheet-menu { 447 | list-style: none; 448 | margin: 0; 449 | padding: 0; 450 | user-select: none; 451 | } 452 | .x-spreadsheet-menu > li { 453 | float: left; 454 | line-height: 1.25em; 455 | padding: 0.785em 1em; 456 | margin: 0; 457 | vertical-align: middle; 458 | text-align: left; 459 | font-weight: 400; 460 | color: #80868b; 461 | white-space: nowrap; 462 | cursor: pointer; 463 | transition: all 0.3s; 464 | font-weight: bold; 465 | } 466 | .x-spreadsheet-menu > li.active { 467 | background-color: #fff; 468 | color: rgba(0, 0, 0, 0.65); 469 | } 470 | .x-spreadsheet-menu > li .x-spreadsheet-icon { 471 | margin: 0 6px; 472 | } 473 | .x-spreadsheet-menu > li .x-spreadsheet-icon .x-spreadsheet-icon-img:hover { 474 | opacity: 0.85; 475 | } 476 | .x-spreadsheet-menu > li .x-spreadsheet-dropdown { 477 | display: inline-block; 478 | } 479 | .x-spreadsheet-toolbar { 480 | border-bottom: 1px solid #e0e2e4; 481 | } 482 | .x-spreadsheet-toolbar .x-spreadsheet-toolbar-btns { 483 | display: inline-flex; 484 | } 485 | .x-spreadsheet-toolbar .x-spreadsheet-toolbar-more { 486 | padding: 0 6px 6px; 487 | text-align: left; 488 | } 489 | .x-spreadsheet-toolbar .x-spreadsheet-toolbar-more .x-spreadsheet-toolbar-divider { 490 | margin-top: 0; 491 | } 492 | .x-spreadsheet-toolbar .x-spreadsheet-toolbar-btn { 493 | flex: 0 0 auto; 494 | display: inline-block; 495 | border: 1px solid transparent; 496 | height: 26px; 497 | line-height: 26px; 498 | min-width: 26px; 499 | margin: 6px 1px 0; 500 | padding: 0; 501 | text-align: center; 502 | border-radius: 2px; 503 | } 504 | .x-spreadsheet-toolbar .x-spreadsheet-toolbar-btn.disabled { 505 | pointer-events: none; 506 | opacity: 0.5; 507 | } 508 | .x-spreadsheet-toolbar .x-spreadsheet-toolbar-btn:hover, 509 | .x-spreadsheet-toolbar .x-spreadsheet-toolbar-btn.active { 510 | background: rgba(0, 0, 0, 0.08); 511 | } 512 | .x-spreadsheet-toolbar-divider { 513 | display: inline-block; 514 | border-right: 1px solid #e0e2e4; 515 | width: 0; 516 | vertical-align: middle; 517 | height: 18px; 518 | margin: 12px 3px 0; 519 | } 520 | .x-spreadsheet-print { 521 | position: absolute; 522 | left: 0; 523 | top: 0; 524 | z-index: 100; 525 | width: 100%; 526 | height: 100%; 527 | display: flex; 528 | flex-direction: column; 529 | } 530 | .x-spreadsheet-print-bar { 531 | background: #424242; 532 | height: 60px; 533 | line-height: 60px; 534 | padding: 0 30px; 535 | } 536 | .x-spreadsheet-print-bar .-title { 537 | color: #fff; 538 | font-weight: bold; 539 | font-size: 1.2em; 540 | float: left; 541 | } 542 | .x-spreadsheet-print-bar .-right { 543 | float: right; 544 | margin-top: 12px; 545 | } 546 | .x-spreadsheet-print-content { 547 | display: flex; 548 | flex: auto; 549 | flex-direction: row; 550 | background: #d0d0d0; 551 | height: calc(100% - 60px); 552 | } 553 | .x-spreadsheet-print-content .-sider { 554 | flex: 0 0 300px; 555 | width: 300px; 556 | border-left: 2px solid #ccc; 557 | background: #fff; 558 | } 559 | .x-spreadsheet-print-content .-content { 560 | flex: auto; 561 | overflow-x: auto; 562 | overflow-y: scroll; 563 | height: 100%; 564 | } 565 | .x-spreadsheet-canvas-card-wraper { 566 | margin: 40px 20px; 567 | } 568 | .x-spreadsheet-canvas-card { 569 | background: #fff; 570 | margin: auto; 571 | page-break-before: auto; 572 | page-break-after: always; 573 | box-shadow: 574 | 0 8px 10px 1px rgba(0, 0, 0, 0.14), 575 | 0 3px 14px 3px rgba(0, 0, 0, 0.12), 576 | 0 4px 5px 0 rgba(0, 0, 0, 0.2); 577 | } 578 | .x-spreadsheet-calendar { 579 | color: rgba(0, 0, 0, 0.65); 580 | background: #ffffff; 581 | user-select: none; 582 | } 583 | .x-spreadsheet-calendar .calendar-header { 584 | font-weight: 700; 585 | line-height: 30px; 586 | text-align: center; 587 | width: 100%; 588 | float: left; 589 | background: #f9fafb; 590 | } 591 | .x-spreadsheet-calendar .calendar-header .calendar-header-left { 592 | padding-left: 5px; 593 | float: left; 594 | } 595 | .x-spreadsheet-calendar .calendar-header .calendar-header-right { 596 | float: right; 597 | } 598 | .x-spreadsheet-calendar .calendar-header .calendar-header-right a { 599 | padding: 3px 0; 600 | margin-right: 2px; 601 | border-radius: 2px; 602 | } 603 | .x-spreadsheet-calendar .calendar-header .calendar-header-right a:hover { 604 | background: rgba(0, 0, 0, 0.08); 605 | } 606 | .x-spreadsheet-calendar .calendar-body { 607 | border-collapse: collapse; 608 | border-spacing: 0; 609 | } 610 | .x-spreadsheet-calendar .calendar-body th, 611 | .x-spreadsheet-calendar .calendar-body td { 612 | width: 100%/7; 613 | min-width: 32px; 614 | text-align: center; 615 | font-weight: 700; 616 | line-height: 30px; 617 | padding: 0; 618 | } 619 | .x-spreadsheet-calendar .calendar-body td > .cell:hover { 620 | background: #ecf6fd; 621 | } 622 | .x-spreadsheet-calendar .calendar-body td > .cell.active, 623 | .x-spreadsheet-calendar .calendar-body td > .cell.active:hover { 624 | background: #ecf6fd; 625 | color: #2185D0; 626 | } 627 | .x-spreadsheet-calendar .calendar-body td > .cell.disabled { 628 | pointer-events: none; 629 | opacity: 0.5; 630 | } 631 | .x-spreadsheet-datepicker { 632 | box-shadow: 2px 2px 5px rgba(0, 0, 0, 0.2); 633 | position: absolute; 634 | left: 0; 635 | top: calc(100% + 5px); 636 | z-index: 10; 637 | width: auto; 638 | } 639 | .x-spreadsheet-buttons { 640 | display: flex; 641 | justify-content: flex-end; 642 | } 643 | .x-spreadsheet-buttons .x-spreadsheet-button { 644 | margin-left: 8px; 645 | } 646 | .x-spreadsheet-button { 647 | display: inline-block; 648 | border-radius: 3px; 649 | line-height: 1em; 650 | min-height: 1em; 651 | white-space: nowrap; 652 | text-align: center; 653 | cursor: pointer; 654 | font-size: 1em; 655 | font-weight: 700; 656 | padding: 0.75em 1em; 657 | color: rgba(0, 0, 0, 0.6); 658 | background: #E0E1E2; 659 | text-decoration: none; 660 | font-family: 661 | "Lato", 662 | "proxima-nova", 663 | "Helvetica Neue", 664 | Arial, 665 | sans-serif; 666 | outline: none; 667 | vertical-align: baseline; 668 | zoom: 1; 669 | user-select: none; 670 | transition: all 0.1s linear; 671 | } 672 | .x-spreadsheet-button.active, 673 | .x-spreadsheet-button:hover { 674 | background-color: #C0C1C2; 675 | color: rgba(0, 0, 0, 0.8); 676 | } 677 | .x-spreadsheet-button.primary { 678 | color: #fff; 679 | background-color: #2185D0; 680 | } 681 | .x-spreadsheet-button.primary:hover, 682 | .x-spreadsheet-button.primary.active { 683 | color: #fff; 684 | background-color: #1678c2; 685 | } 686 | .x-spreadsheet-form-input { 687 | font-size: 1em; 688 | position: relative; 689 | font-weight: 400; 690 | display: inline-flex; 691 | color: rgba(0, 0, 0, 0.87); 692 | } 693 | .x-spreadsheet-form-input input { 694 | z-index: 1; 695 | margin: 0; 696 | max-width: 100%; 697 | flex: 1 0 auto; 698 | outline: 0; 699 | -webkit-tap-highlight-color: rgba(255, 255, 255, 0); 700 | text-align: left; 701 | line-height: 30px; 702 | height: 30px; 703 | padding: 0 8px; 704 | background: #fff; 705 | border: 1px solid #e9e9e9; 706 | border-radius: 3px; 707 | transition: box-shadow 0.1s ease, border-color 0.1s ease; 708 | box-shadow: inset 0 1px 2px hsla(0, 0%, 4%, 0.06); 709 | } 710 | .x-spreadsheet-form-input input:focus { 711 | border-color: #4b89ff; 712 | box-shadow: inset 0 1px 2px rgba(75, 137, 255, 0.2); 713 | } 714 | .x-spreadsheet-form-select { 715 | position: relative; 716 | display: inline-block; 717 | background: #fff; 718 | border: 1px solid #e9e9e9; 719 | border-radius: 2px; 720 | cursor: pointer; 721 | color: rgba(0, 0, 0, 0.87); 722 | user-select: none; 723 | box-shadow: inset 0 1px 2px hsla(0, 0%, 4%, 0.06); 724 | } 725 | .x-spreadsheet-form-select .input-text { 726 | text-overflow: ellipsis; 727 | white-space: nowrap; 728 | min-width: 60px; 729 | width: auto; 730 | height: 30px; 731 | line-height: 30px; 732 | padding: 0 8px; 733 | } 734 | .x-spreadsheet-form-fields { 735 | display: flex; 736 | flex-direction: row; 737 | flex-wrap: wrap; 738 | } 739 | .x-spreadsheet-form-fields .x-spreadsheet-form-field { 740 | flex: 0 1 auto; 741 | } 742 | .x-spreadsheet-form-fields .x-spreadsheet-form-field .label { 743 | display: inline-block; 744 | margin: 0 10px 0 0; 745 | } 746 | .x-spreadsheet-form-field { 747 | display: block; 748 | vertical-align: middle; 749 | margin-left: 10px; 750 | margin-bottom: 10px; 751 | } 752 | .x-spreadsheet-form-field:first-child { 753 | margin-left: 0; 754 | } 755 | .x-spreadsheet-form-field.error .x-spreadsheet-form-select, 756 | .x-spreadsheet-form-field.error input { 757 | border-color: #f04134; 758 | } 759 | .x-spreadsheet-form-field .tip { 760 | color: #f04134; 761 | font-size: 0.9em; 762 | } 763 | .x-spreadsheet-dimmer { 764 | display: none; 765 | position: absolute; 766 | top: 0 !important; 767 | left: 0 !important; 768 | width: 100%; 769 | height: 100%; 770 | text-align: center; 771 | vertical-align: middle; 772 | background-color: rgba(0, 0, 0, 0.6); 773 | opacity: 0; 774 | -webkit-animation-fill-mode: both; 775 | animation-fill-mode: both; 776 | -webkit-animation-duration: 0.5s; 777 | animation-duration: 0.5s; 778 | transition: background-color 0.5s linear; 779 | user-select: none; 780 | z-index: 1000; 781 | } 782 | .x-spreadsheet-dimmer.active { 783 | display: block; 784 | opacity: 1; 785 | } 786 | form fieldset { 787 | border: none; 788 | } 789 | form fieldset label { 790 | display: block; 791 | margin-bottom: 0.5em; 792 | font-size: 1em; 793 | color: #666; 794 | } 795 | form fieldset select { 796 | font-size: 1.1em; 797 | width: 100%; 798 | background-color: #fff; 799 | border: none; 800 | border-bottom: 2px solid #ddd; 801 | padding: 0.5em 0.85em; 802 | border-radius: 2px; 803 | } 804 | .x-spreadsheet-modal, 805 | .x-spreadsheet-toast { 806 | font-size: 13px; 807 | position: fixed; 808 | z-index: 1001; 809 | text-align: left; 810 | line-height: 1.25em; 811 | min-width: 360px; 812 | color: rgba(0, 0, 0, 0.87); 813 | font-family: 814 | "Lato", 815 | "Source Sans Pro", 816 | Roboto, 817 | Helvetica, 818 | Arial, 819 | sans-serif; 820 | border-radius: 4px; 821 | border: 1px solid rgba(0, 0, 0, 0.1); 822 | background-color: #fff; 823 | background-clip: padding-box; 824 | box-shadow: rgba(0, 0, 0, 0.2) 0px 2px 8px; 825 | } 826 | .x-spreadsheet-toast { 827 | background-color: rgba(255, 255, 255, 0.85); 828 | } 829 | .x-spreadsheet-modal-header, 830 | .x-spreadsheet-toast-header { 831 | font-weight: 600; 832 | background-clip: padding-box; 833 | background-color: rgba(255, 255, 255, 0.85); 834 | border-bottom: 1px solid rgba(0, 0, 0, 0.05); 835 | border-radius: 4px 4px 0 0; 836 | } 837 | .x-spreadsheet-modal-header .x-spreadsheet-icon, 838 | .x-spreadsheet-toast-header .x-spreadsheet-icon { 839 | position: absolute; 840 | right: 0.8em; 841 | top: 0.65em; 842 | border-radius: 18px; 843 | } 844 | .x-spreadsheet-modal-header .x-spreadsheet-icon:hover, 845 | .x-spreadsheet-toast-header .x-spreadsheet-icon:hover { 846 | opacity: 1; 847 | background: rgba(0, 0, 0, 0.08); 848 | } 849 | .x-spreadsheet-toast-header { 850 | color: #F2711C; 851 | } 852 | .x-spreadsheet-modal-header { 853 | border-bottom: 1px solid #e0e2e4; 854 | background: rgba(0, 0, 0, 0.08); 855 | font-size: 1.0785em; 856 | } 857 | .x-spreadsheet-modal-header, 858 | .x-spreadsheet-modal-content, 859 | .x-spreadsheet-toast-header, 860 | .x-spreadsheet-toast-content { 861 | padding: 0.75em 1em; 862 | } 863 | @media screen and (min-width: 320px) and (max-width: 480px) { 864 | .x-spreadsheet-toolbar { 865 | display: none; 866 | } 867 | } 868 | .x-spreadsheet-icon { 869 | width: 18px; 870 | height: 18px; 871 | margin: 1px 1px 2px 1px; 872 | text-align: center; 873 | vertical-align: middle; 874 | user-select: none; 875 | overflow: hidden; 876 | position: relative; 877 | display: inline-block; 878 | } 879 | .x-spreadsheet-icon .x-spreadsheet-icon-img { 880 | background-image: url(data:image/svg+xml;base64,<?xml version='1.0' encoding='UTF-8'?><!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"><svg xmlns:xlink="http://www.w3.org/1999/xlink" xmlns="http://www.w3.org/2000/svg" version="1.1" x="0" y="0" width="262px" height="72px" viewBox="0 0 262 72" preserveAspectRatio="none">
<g xmlns="http://www.w3.org/2000/svg" transform="translate(0,0)"><path fill="#000000" fill-rule="evenodd" d="M11.5656391,4.43436088 L9,7 L16,7 L16,0 L13.0418424,2.95815758 C11.5936787,1.73635959 9.72260775,1 7.67955083,1 C4.22126258,1 1.25575599,3.10984908 0,6 L2,7 C2.93658775,4.60974406 5.12943697,3.08011229 7.67955083,3 C9.14881247,3.0528747 10.4994783,3.57862053 11.5656391,4.43436088 Z" transform="matrix(-1 0 0 1 17 5)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(18,0)"><path fill="#000000" fill-rule="evenodd" d="M11.5656391,4.43436088 L9,7 L16,7 L16,0 L13.0418424,2.95815758 C11.5936787,1.73635959 9.72260775,1 7.67955083,1 C4.22126258,1 1.25575599,3.10984908 0,6 L2,7 C2.93658775,4.60974406 5.12943697,3.08011229 7.67955083,3 C9.14881247,3.0528747 10.4994783,3.57862053 11.5656391,4.43436088 Z" transform="translate(1 5)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(36,0)"><path fill="#000000" fill-rule="evenodd" d="M13,14 L3,14 L3,11 L0,11 L0,6.00591905 C0,4.89808055 0.894513756,4 1.99406028,4 L14.0059397,4 C15.1072288,4 16,4.88655484 16,6.00591905 L16,11 L13,11 L13,14 Z M5,9 L11,9 L11,12 L5,12 L5,9 Z M3,0 L13,0 L13,3 L3,3 L3,0 Z M12,6 L14,6 L14,8 L12,8 L12,6 Z" transform="translate(1 2)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(54,0)"><path fill="#000000" fill-rule="evenodd" d="M9,0 L1,0 C0.45,0 0,0.45 0,1 L0,4 C0,4.55 0.45,5 1,5 L9,5 C9.55,5 10,4.55 10,4 L10,3 L11,3 L11,6 L4,6 L4,14 L6,14 L6,8 L13,8 L13,2 L10,2 L10,1 C10,0.45 9.55,0 9,0 Z" transform="translate(3 2)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(72,0)"><path fill="#000000" fill-rule="evenodd" d="M0.27,1.55 L5.43,6.7 L3,12 L5.5,12 L7.14,8.42 L11.73,13 L13,11.73 L1.55,0.27 L0.27,1.55 L0.27,1.55 Z M3.82,0 L5.82,2 L7.58,2 L7.03,3.21 L8.74,4.92 L10.08,2 L14,2 L14,0 L3.82,0 L3.82,0 Z" transform="translate(2 3)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(90,0)"><path fill="#000000" fill-rule="evenodd" d="M9,3.5 C9,1.57 7.43,0 5.5,0 L1.77635684e-15,0 L1.77635684e-15,12 L6.25,12 C8.04,12 9.5,10.54 9.5,8.75 C9.5,7.45 8.73,6.34 7.63,5.82 C8.46,5.24 9,4.38 9,3.5 Z M5,2 C5.82999992,2 6.5,2.67 6.5,3.5 C6.5,4.33 5.82999992,5 5,5 L3,5 L3,2 L5,2 Z M3,10 L3,7 L5.5,7 C6.32999992,7 7,7.67 7,8.5 C7,9.33 6.32999992,10 5.5,10 L3,10 Z" transform="translate(4 3)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(108,0)"><polygon fill="#000000" fill-rule="evenodd" points="4 0 4 2 6.58 2 2.92 10 0 10 0 12 8 12 8 10 5.42 10 9.08 2 12 2 12 0" transform="translate(3 3)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(126,0)"><path fill="#000000" d="M6,12 C8.76,12 11,9.76 11,7 L11,0 L9,0 L9,7 C9,8.75029916 7.49912807,10 6,10 C4.50087193,10 3,8.75837486 3,7 L3,0 L1,0 L1,7 C1,9.76 3.24,12 6,12 Z M0,13 L0,15 L12,15 L12,13 L0,13 Z" transform="translate(3 3)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(144,0)"><path fill="#010101" fill-rule="evenodd" d="M2.8875,3.06 C2.8875,2.6025 2.985,2.18625 3.18375,1.8075 C3.3825,1.42875 3.66,1.10625 4.02,0.84 C4.38,0.57375 4.80375,0.3675 5.29875,0.22125 C5.79375,0.075 6.33375,0 6.92625,0 C7.53375,0 8.085,0.0825 8.58,0.25125 C9.075,0.42 9.49875,0.6525 9.85125,0.95625 C10.20375,1.25625 10.47375,1.6125 10.665,2.02875 C10.85625,2.44125 10.95,2.895 10.95,3.38625 L8.6925,3.38625 C8.6925,3.1575 8.655,2.94375 8.58375,2.74875 C8.5125,2.55 8.4,2.38125 8.25,2.2425 C8.1,2.10375 7.9125,1.99125 7.6875,1.91625 C7.4625,1.8375 7.19625,1.8 6.88875,1.8 C6.5925,1.8 6.3375,1.83375 6.11625,1.8975 C5.89875,1.96125 5.71875,2.05125 5.57625,2.1675 C5.43375,2.28375 5.325,2.41875 5.25375,2.5725 C5.1825,2.72625 5.145,2.895 5.145,3.0675 C5.145,3.4275 5.32875,3.73125 5.69625,3.975 C5.71780203,3.98908066 5.73942012,4.00311728 5.76118357,4.01733315 C6.02342923,4.18863185 6.5,4.5 7,5 L4,5 C4,5 3.21375,4.37625 3.17625,4.30875 C2.985,3.9525 2.8875,3.53625 2.8875,3.06 Z M14,6 L0,6 L0,8 L7.21875,8 C7.35375,8.0525 7.51875,8.105 7.63125,8.15375 C7.90875,8.2775 8.12625,8.40875 8.28375,8.53625 C8.44125,8.6675 8.54625,8.81 8.6025,8.96 C8.65875,9.11375 8.685,9.28625 8.685,9.47375 C8.685,9.65 8.65125,9.815 8.58375,9.965 C8.51625,10.11875 8.41125,10.25 8.2725,10.35875 C8.13375,10.4675 7.95375,10.55375 7.74,10.6175 C7.5225,10.68125 7.27125,10.71125 6.97875,10.71125 C6.6525,10.71125 6.35625,10.6775 6.09,10.61375 C5.82375,10.55 5.59875,10.445 5.41125,10.3025 C5.22375,10.16 5.0775,9.9725 4.9725,9.74375 C4.8675,9.515 4.78125,9.17 4.78125,9 L2.55,9 C2.55,9.2525 2.61,9.6875 2.72625,10.025 C2.8425,10.3625 3.0075,10.66625 3.21375,10.9325 C3.42,11.19875 3.6675,11.4275 3.94875,11.6225 C4.23,11.8175 4.53375,11.9825 4.86375,12.11 C5.19375,12.24125 5.535,12.33875 5.89875,12.39875 C6.25875,12.4625 6.6225,12.4925 6.9825,12.4925 C7.5825,12.4925 8.13,12.425 8.6175,12.28625 C9.105,12.1475 9.525,11.94875 9.87,11.69375 C10.215,11.435 10.48125,11.12 10.6725,10.74125 C10.86375,10.3625 10.95375,9.935 10.95375,9.455 C10.95375,9.005 10.875,8.6 10.72125,8.24375 C10.68375,8.1575 10.6425,8.075 10.59375,7.9925 L14,8 L14,6 Z" transform="translate(2 3)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(162,0)"><path fill="#000000" fill-rule="evenodd" d="M7,0 L5,0 L0.5,12 L2.5,12 L3.62,9 L8.37,9 L9.49,12 L11.49,12 L7,0 L7,0 Z M4.38,7 L6,2.67 L7.62,7 L4.38,7 L4.38,7 Z" transform="translate(3 1)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(180,0)"><g fill="none" fill-rule="evenodd">
    <path fill="#000000" d="M14.5,8.87 C14.5,8.87 13,10.49 13,11.49 C13,12.32 13.67,12.99 14.5,12.99 C15.33,12.99 16,12.32 16,11.49 C16,10.5 14.5,8.87 14.5,8.87 L14.5,8.87 Z M12.71,6.79 L5.91,0 L4.85,1.06 L6.44,2.65 L2.29,6.79 C1.9,7.18 1.9,7.81 2.29,8.2 L6.79,12.7 C6.99,12.9 7.24,13 7.5,13 C7.76,13 8.01,12.9 8.21,12.71 L12.71,8.21 C13.1,7.82 13.1,7.18 12.71,6.79 L12.71,6.79 Z M4.21,7 L7.5,3.71 L10.79,7 L4.21,7 L4.21,7 Z"/>
  </g>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(198,0)"><path fill="#000000" fill-rule="evenodd" d="M3,6 L1,6 L1,2 L8,2 L8,4 L3,4 L3,6 Z M10,4 L10,2 L17,2 L17,6 L15,6 L15,4 L10,4 Z M10,14 L15,14 L15,12 L17,12 L17,16 L10,16 L10,14 Z M1,12 L3,12 L3,14 L8,14 L8,16 L1,16 L1,12 Z M1,8 L5,8 L5,6 L8,9 L5,12 L5,10 L1,10 L1,8 Z M10,9 L13,6 L13,8 L17,8 L17,10 L13,10 L13,12 L10,9 Z"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(216,0)"><path fill="#000000" fill-rule="evenodd" d="M0,14 L10,14 L10,12 L0,12 L0,14 Z M10,4 L0,4 L0,6 L10,6 L10,4 Z M0,0 L0,2 L14,2 L14,0 L0,0 Z M0,10 L14,10 L14,8 L0,8 L0,10 Z" transform="translate(2 2)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(234,0)"><path fill="#000000" fill-rule="evenodd" d="M2,12 L2,14 L12,14 L12,12 L2,12 Z M2,4 L2,6 L12,6 L12,4 L2,4 Z M0,10 L14,10 L14,8 L0,8 L0,10 Z M0,0 L0,2 L14,2 L14,0 L0,0 Z" transform="translate(2 2)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(0,18)"><path fill="#000000" fill-rule="evenodd" d="M4,14 L14,14 L14,12 L4,12 L4,14 Z M0,10 L14,10 L14,8 L0,8 L0,10 Z M0,0 L0,2 L14,2 L14,0 L0,0 Z M4,6 L14,6 L14,4 L4,4 L4,6 Z" transform="translate(2 2)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(18,18)"><path fill="#000000" fill-rule="evenodd" d="M0,0 L0,2 L12,2 L12,0 L0,0 L0,0 Z M2.5,7 L5,7 L5,14 L7,14 L7,7 L9.5,7 L6,3.5 L2.5,7 L2.5,7 Z" transform="translate(3 2)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(36,18)"><path fill="#000000" fill-rule="evenodd" d="M9.5,3 L7,3 L7,0 L5,0 L5,3 L2.5,3 L6,6.5 L9.5,3 L9.5,3 Z M0,8 L0,10 L12,10 L12,8 L0,8 L0,8 Z M2.5,15 L5,15 L5,18 L7,18 L7,15 L9.5,15 L6,11.5 L2.5,15 L2.5,15 Z" transform="translate(3)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(54,18)"><path fill="#000000" fill-rule="evenodd" d="M9.5,7 L7,7 L7,0 L5,0 L5,7 L2.5,7 L6,10.5 L9.5,7 L9.5,7 Z M0,12 L0,14 L12,14 L12,12 L0,12 L0,12 Z" transform="translate(3 2)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(72,18)"><path fill="#000000" fill-rule="evenodd" d="M14,0 L0,0 L0,2 L14,2 L14,0 Z M0,12 L4,12 L4,10 L0,10 L0,12 Z M11.5,5 L0,5 L0,7 L11.75,7 C12.58,7 13.25,7.67 13.25,8.5 C13.25,9.33 12.58,10 11.75,10 L9,10 L9,8 L6,11 L9,14 L9,12 L11.5,12 C13.43,12 15,10.43 15,8.5 C15,6.57 13.43,5 11.5,5 Z" transform="translate(2 3)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(90,18)"><path fill="#000000" fill-rule="evenodd" d="M0,0 L0,1 L6,7 L6,12 L8,11 L8,7 L14,1 L14,0 L0,0 Z M4,3 L10,3 L7,6 L4,3 Z" transform="translate(2 3)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(108,18)"><polygon fill="#000000" fill-rule="evenodd" points="10 0 0 0 0 1.8 5.5 7 0 12.2 0 14 10 14 10 12 3.1 12 8 7 3.1 2 10 2" transform="translate(4 2)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(126,18)"><polygon fill="#000000" fill-rule="evenodd" points="0 0 4 4 8 0" transform="translate(5 7)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(144,18)"><polygon fill="#000000" fill-rule="evenodd" points="-2 2 2 6 6 2" transform="rotate(-90 8 3)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(162,18)"><path fill="#000000" fill-rule="evenodd" d="M1.9,4 C1.9,2.84 2.84,1.9 4,1.9 L8,1.9 L8,0 L4,0 C1.79,0 0,1.79 0,4 C0,6.21 1.79,8 4,8 L8,8 L8,6.1 L4,6.1 C2.84,6.1 1.9,5.16 1.9,4 L1.9,4 Z M14,0 L10,0 L10,1.9 L14,1.9 C15.16,1.9 16.1,2.84 16.1,4 C16.1,5.16 15.16,6.1 14,6.1 L10,6.1 L10,8 L14,8 C16.21,8 18,6.21 18,4 C18,1.79 16.21,0 14,0 L14,0 Z M6,5 L12,5 L12,3 L6,3 L6,5 L6,5 Z" transform="translate(0 5)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(180,18)"><path fill="#000000" fill-rule="evenodd" d="M15,0 C15.55,0 16,0.45 16,1 L16,15 C16,15.55 15.55,16 15,16 L1,16 C0.45,16 0,15.55 0,15 L0,1 C0,0.45 0.45,0 1,0 L15,0 Z M2,2 L2,14 L14,14 L14,2 L2,2 Z M6,12 L4,12 L4,7 L6,7 L6,12 L6,12 Z M9,12 L7,12 L7,4 L9,4 L9,12 L9,12 Z M12,12 L10,12 L10,8 L12,8 L12,12 L12,12 Z" transform="translate(1 1)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(198,18)"><g fill="none" fill-rule="evenodd">
    <path stroke="#000000" d="M1.5 3.5H16.5V15.5H1.5z"/>
    <path fill="#000000" d="M6 8H7V15H6z"/>
    <path fill="#D8D8D8" d="M2 4H16V7H2z"/>
    <path fill="#000000" d="M2 7H16V8H2zM2 11H16V12H2z"/>
  </g>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(216,18)"><path fill="#000000" fill-rule="evenodd" d="M2,0.5 C1.17,0.5 0.5,1.17 0.5,2 C0.5,2.83 1.17,3.5 2,3.5 C2.83,3.5 3.5,2.83 3.5,2 C3.5,1.17 2.83,0.5 2,0.5 L2,0.5 Z M12,0.5 C11.17,0.5 10.5,1.17 10.5,2 C10.5,2.83 11.17,3.5 12,3.5 C12.83,3.5 13.5,2.83 13.5,2 C13.5,1.17 12.83,0.5 12,0.5 L12,0.5 Z M7,0.5 C6.17,0.5 5.5,1.17 5.5,2 C5.5,2.83 6.17,3.5 7,3.5 C7.83,3.5 8.5,2.83 8.5,2 C8.5,1.17 7.83,0.5 7,0.5 L7,0.5 Z" transform="translate(2 7)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(234,18)"><path fill="#000000" fill-rule="evenodd" d="M6,4 L6,0 L4,0 L4,4 L0,4 L0,6 L4,6 L4,10 L6,10 L6,6 L10,6 L10,4 L6,4 Z" transform="translate(4 4)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(0,36)"><path fill="#000000" fill-rule="evenodd" d="M0,0 L0,14 L14,14 L14,0 L0,0 L0,0 Z M6,12 L2,12 L2,8 L6,8 L6,12 L6,12 Z M6,6 L2,6 L2,2 L6,2 L6,6 L6,6 Z M12,12 L8,12 L8,8 L12,8 L12,12 L12,12 Z M12,6 L8,6 L8,2 L12,2 L12,6 L12,6 Z" transform="translate(2 2)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(18,36)">
  <g fill="#000000" fill-rule="evenodd" transform="translate(2 2)">
    <path d="M0,14 L2,14 L2,12 L0,12 L0,14 L0,14 Z M2,3 L0,3 L0,5 L2,5 L2,3 L2,3 Z M3,14 L5,14 L5,12 L3,12 L3,14 L3,14 Z M11,0 L9,0 L9,2 L11,2 L11,0 L11,0 Z M2,0 L0,0 L0,2 L2,2 L2,0 L2,0 Z M5,0 L3,0 L3,2 L5,2 L5,0 L5,0 Z M0,11 L2,11 L2,9 L0,9 L0,11 L0,11 Z M9,14 L11,14 L11,12 L9,12 L9,14 L9,14 Z M12,0 L12,2 L14,2 L14,0 L12,0 L12,0 Z M12,5 L14,5 L14,3 L12,3 L12,5 L12,5 Z M12,14 L14,14 L14,12 L12,12 L12,14 L12,14 Z M12,11 L14,11 L14,9 L12,9 L12,11 L12,11 Z" opacity=".54"/>
    <polygon points="8 0 6 0 6 6 0 6 0 8 6 8 6 14 8 14 8 8 14 8 14 6 8 6"/>
  </g>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(36,36)">
  <g fill="#000000" fill-rule="evenodd" transform="translate(2 2)">
    <path d="M6,14 L8,14 L8,12 L6,12 L6,14 L6,14 Z M3,2 L5,2 L5,0 L3,0 L3,2 L3,2 Z M6,11 L8,11 L8,9 L6,9 L6,11 L6,11 Z M3,14 L5,14 L5,12 L3,12 L3,14 L3,14 Z M0,5 L2,5 L2,3 L0,3 L0,5 L0,5 Z M0,14 L2,14 L2,12 L0,12 L0,14 L0,14 Z M0,2 L2,2 L2,0 L0,0 L0,2 L0,2 Z M0,11 L2,11 L2,9 L0,9 L0,11 L0,11 Z M12,11 L14,11 L14,9 L12,9 L12,11 L12,11 Z M12,14 L14,14 L14,12 L12,12 L12,14 L12,14 Z M12,5 L14,5 L14,3 L12,3 L12,5 L12,5 Z M12,0 L12,2 L14,2 L14,0 L12,0 L12,0 Z M6,2 L8,2 L8,0 L6,0 L6,2 L6,2 Z M9,2 L11,2 L11,0 L9,0 L9,2 L9,2 Z M6,5 L8,5 L8,3 L6,3 L6,5 L6,5 Z M9,14 L11,14 L11,12 L9,12 L9,14 L9,14 Z" opacity=".54"/>
    <polygon points="0 8 14 8 14 6 0 6"/>
  </g>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(54,36)">
  <g fill="#000000" fill-rule="evenodd" transform="translate(2 2)">
    <path d="M3,14 L5,14 L5,12 L3,12 L3,14 L3,14 Z M0,5 L2,5 L2,3 L0,3 L0,5 L0,5 Z M0,2 L2,2 L2,0 L0,0 L0,2 L0,2 Z M3,8 L5,8 L5,6 L3,6 L3,8 L3,8 Z M3,2 L5,2 L5,0 L3,0 L3,2 L3,2 Z M0,14 L2,14 L2,12 L0,12 L0,14 L0,14 Z M0,8 L2,8 L2,6 L0,6 L0,8 L0,8 Z M0,11 L2,11 L2,9 L0,9 L0,11 L0,11 Z M12,0 L12,2 L14,2 L14,0 L12,0 L12,0 Z M12,8 L14,8 L14,6 L12,6 L12,8 L12,8 Z M12,14 L14,14 L14,12 L12,12 L12,14 L12,14 Z M12,5 L14,5 L14,3 L12,3 L12,5 L12,5 Z M12,11 L14,11 L14,9 L12,9 L12,11 L12,11 Z M9,14 L11,14 L11,12 L9,12 L9,14 L9,14 Z M9,8 L11,8 L11,6 L9,6 L9,8 L9,8 Z M9,2 L11,2 L11,0 L9,0 L9,2 L9,2 Z" opacity=".54"/>
    <polygon points="6 14 8 14 8 0 6 0"/>
  </g>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(72,36)">
  <g fill="#000000" fill-rule="evenodd" transform="translate(2 2)">
    <path d="M8,3 L6,3 L6,5 L8,5 L8,3 L8,3 Z M11,6 L9,6 L9,8 L11,8 L11,6 L11,6 Z M8,6 L6,6 L6,8 L8,8 L8,6 L8,6 Z M8,9 L6,9 L6,11 L8,11 L8,9 L8,9 Z M5,6 L3,6 L3,8 L5,8 L5,6 L5,6 Z" opacity=".54"/>
    <path d="M0,0 L14,0 L14,14 L0,14 L0,0 Z M12,12 L12,2 L2,2 L2,12 L12,12 Z"/>
  </g>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(90,36)">
  <g fill="#000000" fill-rule="evenodd" transform="translate(2 2)">
    <path d="M6,8 L8,8 L8,6 L6,6 L6,8 L6,8 Z M6,5 L8,5 L8,3 L6,3 L6,5 L6,5 Z M6,11 L8,11 L8,9 L6,9 L6,11 L6,11 Z M6,14 L8,14 L8,12 L6,12 L6,14 L6,14 Z M3,14 L5,14 L5,12 L3,12 L3,14 L3,14 Z M3,2 L5,2 L5,0 L3,0 L3,2 L3,2 Z M3,8 L5,8 L5,6 L3,6 L3,8 L3,8 Z M12,14 L14,14 L14,12 L12,12 L12,14 L12,14 Z M12,8 L14,8 L14,6 L12,6 L12,8 L12,8 Z M12,11 L14,11 L14,9 L12,9 L12,11 L12,11 Z M12,5 L14,5 L14,3 L12,3 L12,5 L12,5 Z M6,2 L8,2 L8,0 L6,0 L6,2 L6,2 Z M12,0 L12,2 L14,2 L14,0 L12,0 L12,0 Z M9,14 L11,14 L11,12 L9,12 L9,14 L9,14 Z M9,8 L11,8 L11,6 L9,6 L9,8 L9,8 Z M9,2 L11,2 L11,0 L9,0 L9,2 L9,2 Z" opacity=".54"/>
    <polygon points="0 14 2 14 2 0 0 0"/>
  </g>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(108,36)">
  <g fill="#000000" fill-rule="evenodd" transform="translate(2 2)">
    <path d="M3,8 L5,8 L5,6 L3,6 L3,8 L3,8 Z M0,14 L2,14 L2,12 L0,12 L0,14 L0,14 Z M6,14 L8,14 L8,12 L6,12 L6,14 L6,14 Z M6,11 L8,11 L8,9 L6,9 L6,11 L6,11 Z M3,14 L5,14 L5,12 L3,12 L3,14 L3,14 Z M0,11 L2,11 L2,9 L0,9 L0,11 L0,11 Z M6,8 L8,8 L8,6 L6,6 L6,8 L6,8 Z M0,5 L2,5 L2,3 L0,3 L0,5 L0,5 Z M0,8 L2,8 L2,6 L0,6 L0,8 L0,8 Z M12,8 L14,8 L14,6 L12,6 L12,8 L12,8 Z M12,11 L14,11 L14,9 L12,9 L12,11 L12,11 Z M12,5 L14,5 L14,3 L12,3 L12,5 L12,5 Z M6,5 L8,5 L8,3 L6,3 L6,5 L6,5 Z M9,14 L11,14 L11,12 L9,12 L9,14 L9,14 Z M9,8 L11,8 L11,6 L9,6 L9,8 L9,8 Z M12,14 L14,14 L14,12 L12,12 L12,14 L12,14 Z" opacity=".54"/>
    <polygon points="0 0 0 2 14 2 14 0"/>
  </g>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(126,36)">
  <g fill="#000000" fill-rule="evenodd" transform="translate(2 2)">
    <path d="M0,2 L2,2 L2,0 L0,0 L0,2 L0,2 Z M3,2 L5,2 L5,0 L3,0 L3,2 L3,2 Z M3,8 L5,8 L5,6 L3,6 L3,8 L3,8 Z M3,14 L5,14 L5,12 L3,12 L3,14 L3,14 Z M0,5 L2,5 L2,3 L0,3 L0,5 L0,5 Z M0,8 L2,8 L2,6 L0,6 L0,8 L0,8 Z M0,14 L2,14 L2,12 L0,12 L0,14 L0,14 Z M0,11 L2,11 L2,9 L0,9 L0,11 L0,11 Z M9,8 L11,8 L11,6 L9,6 L9,8 L9,8 Z M6,14 L8,14 L8,12 L6,12 L6,14 L6,14 Z M9,14 L11,14 L11,12 L9,12 L9,14 L9,14 Z M6,2 L8,2 L8,0 L6,0 L6,2 L6,2 Z M9,2 L11,2 L11,0 L9,0 L9,2 L9,2 Z M6,11 L8,11 L8,9 L6,9 L6,11 L6,11 Z M6,5 L8,5 L8,3 L6,3 L6,5 L6,5 Z M6,8 L8,8 L8,6 L6,6 L6,8 L6,8 Z" opacity=".54"/>
    <polygon points="12 0 12 14 14 14 14 0"/>
  </g>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(144,36)">
  <g fill="#000000" fill-rule="evenodd" transform="translate(2 2)">
    <path d="M5,0 L3,0 L3,2 L5,2 L5,0 L5,0 Z M8,6 L6,6 L6,8 L8,8 L8,6 L8,6 Z M8,9 L6,9 L6,11 L8,11 L8,9 L8,9 Z M11,6 L9,6 L9,8 L11,8 L11,6 L11,6 Z M5,6 L3,6 L3,8 L5,8 L5,6 L5,6 Z M11,0 L9,0 L9,2 L11,2 L11,0 L11,0 Z M8,3 L6,3 L6,5 L8,5 L8,3 L8,3 Z M8,0 L6,0 L6,2 L8,2 L8,0 L8,0 Z M2,9 L0,9 L0,11 L2,11 L2,9 L2,9 Z M12,11 L14,11 L14,9 L12,9 L12,11 L12,11 Z M12,5 L14,5 L14,3 L12,3 L12,5 L12,5 Z M12,8 L14,8 L14,6 L12,6 L12,8 L12,8 Z M12,0 L12,2 L14,2 L14,0 L12,0 L12,0 Z M2,0 L0,0 L0,2 L2,2 L2,0 L2,0 Z M2,3 L0,3 L0,5 L2,5 L2,3 L2,3 Z M2,6 L0,6 L0,8 L2,8 L2,6 L2,6 Z" opacity=".54"/>
    <polygon points="0 14 14 14 14 12 0 12"/>
  </g>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(162,36)"><path fill="#000000" fill-rule="evenodd" d="M6,14 L8,14 L8,12 L6,12 L6,14 L6,14 Z M3,8 L5,8 L5,6 L3,6 L3,8 L3,8 Z M3,2 L5,2 L5,0 L3,0 L3,2 L3,2 Z M6,11 L8,11 L8,9 L6,9 L6,11 L6,11 Z M3,14 L5,14 L5,12 L3,12 L3,14 L3,14 Z M0,5 L2,5 L2,3 L0,3 L0,5 L0,5 Z M0,14 L2,14 L2,12 L0,12 L0,14 L0,14 Z M0,2 L2,2 L2,0 L0,0 L0,2 L0,2 Z M0,8 L2,8 L2,6 L0,6 L0,8 L0,8 Z M6,8 L8,8 L8,6 L6,6 L6,8 L6,8 Z M0,11 L2,11 L2,9 L0,9 L0,11 L0,11 Z M12,11 L14,11 L14,9 L12,9 L12,11 L12,11 Z M12,14 L14,14 L14,12 L12,12 L12,14 L12,14 Z M12,8 L14,8 L14,6 L12,6 L12,8 L12,8 Z M12,5 L14,5 L14,3 L12,3 L12,5 L12,5 Z M12,0 L12,2 L14,2 L14,0 L12,0 L12,0 Z M6,2 L8,2 L8,0 L6,0 L6,2 L6,2 Z M9,2 L11,2 L11,0 L9,0 L9,2 L9,2 Z M6,5 L8,5 L8,3 L6,3 L6,5 L6,5 Z M9,14 L11,14 L11,12 L9,12 L9,14 L9,14 Z M9,8 L11,8 L11,6 L9,6 L9,8 L9,8 Z" transform="translate(2 2)" opacity=".54"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(180,36)"><path fill="#000000" fill-rule="evenodd" d="M6.5,3.62 L0,10.12 L0,13 L2.88,13 L9.38,6.5 L6.5,3.62 Z M11.85,4.02 C12.05,3.82 12.05,3.51 11.85,3.31 L9.68,1.14 C9.48,0.94 9.17,0.94 8.97,1.14 L7.62,2.5 L10.5,5.38 L11.85,4.02 L11.85,4.02 Z" transform="translate(4)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(198,36)"><path fill="#000000" fill-rule="evenodd" d="M0,0 L14,0 L14,2 L0,2 L0,0 Z M0,4 L6,4 L6,6 L0,6 L0,4 Z M0,8 L2,8 L2,10 L0,10 L0,8 Z M8,4 L14,4 L14,6 L8,6 L8,4 Z M4,8 L6,8 L6,10 L4,10 L4,8 Z M8,8 L10,8 L10,10 L8,10 L8,8 Z M12,8 L14,8 L14,10 L12,10 L12,8 Z" transform="translate(2 4)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(216,36)"><polygon fill="#000000" fill-rule="evenodd" points="-2 2 2 6 6 2" transform="rotate(90 3 10)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(234,36)"><polygon fill="#000000" fill-rule="evenodd" points="7.53 1.53 6.47 .47 4 2.94 1.53 .47 .47 1.53 2.94 4 .47 6.47 1.53 7.53 4 5.06 6.47 7.53 7.53 6.47 5.06 4" transform="translate(5 5)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(0,54)"><polygon fill="#000000" fill-rule="evenodd" points="8.44 .44 5 3.88 1.56 .44 .5 1.5 5 6 9.5 1.5" transform="translate(4 6)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(18,54)"><polygon fill="#000000" fill-rule="evenodd" points="5 0 .5 4.5 1.56 5.56 5 2.12 8.44 5.56 9.5 4.5" transform="translate(4 6)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(36,54)"><polygon fill="#000000" fill-rule="evenodd" points="8.44 .44 5 3.88 1.56 .44 .5 1.5 5 6 9.5 1.5" transform="rotate(90 4 8)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(54,54)"><polygon fill="#000000" fill-rule="evenodd" points="5 0 .5 4.5 1.56 5.56 5 2.12 8.44 5.56 9.5 4.5" transform="rotate(90 4 8)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(72,54)"><polygon fill="#000000" fill-rule="evenodd" points="12 5 3.125 5 7.06 1.06 6 0 0 6 6 12 7.06 10.94 3.125 7 12 7" transform="matrix(-1 0 0 1 15 3)"/>
</g>
<g xmlns="http://www.w3.org/2000/svg" transform="translate(90,54)"><polygon fill="#000000" fill-rule="evenodd" points="12 5 3.125 5 7.06 1.06 6 0 0 6 6 12 7.06 10.94 3.125 7 12 7" transform="translate(3 3)"/>
</g>
</svg>
); 881 | position: absolute; 882 | width: 262px; 883 | height: 444px; 884 | opacity: 0.56; 885 | } 886 | .x-spreadsheet-icon .x-spreadsheet-icon-img.undo { 887 | left: 0; 888 | top: 0; 889 | } 890 | .x-spreadsheet-icon .x-spreadsheet-icon-img.redo { 891 | left: -18px; 892 | top: 0; 893 | } 894 | .x-spreadsheet-icon .x-spreadsheet-icon-img.print { 895 | left: -36px; 896 | top: 0; 897 | } 898 | .x-spreadsheet-icon .x-spreadsheet-icon-img.paintformat { 899 | left: -54px; 900 | top: 0; 901 | } 902 | .x-spreadsheet-icon .x-spreadsheet-icon-img.clearformat { 903 | left: -72px; 904 | top: 0; 905 | } 906 | .x-spreadsheet-icon .x-spreadsheet-icon-img.font-bold { 907 | left: -90px; 908 | top: 0; 909 | } 910 | .x-spreadsheet-icon .x-spreadsheet-icon-img.font-italic { 911 | left: -108px; 912 | top: 0; 913 | } 914 | .x-spreadsheet-icon .x-spreadsheet-icon-img.underline { 915 | left: -126px; 916 | top: 0; 917 | } 918 | .x-spreadsheet-icon .x-spreadsheet-icon-img.strike { 919 | left: -144px; 920 | top: 0; 921 | } 922 | .x-spreadsheet-icon .x-spreadsheet-icon-img.color { 923 | left: -162px; 924 | top: 0; 925 | } 926 | .x-spreadsheet-icon .x-spreadsheet-icon-img.bgcolor { 927 | left: -180px; 928 | top: 0; 929 | } 930 | .x-spreadsheet-icon .x-spreadsheet-icon-img.merge { 931 | left: -198px; 932 | top: 0; 933 | } 934 | .x-spreadsheet-icon .x-spreadsheet-icon-img.align-left { 935 | left: -216px; 936 | top: 0; 937 | } 938 | .x-spreadsheet-icon .x-spreadsheet-icon-img.align-center { 939 | left: -234px; 940 | top: 0; 941 | } 942 | .x-spreadsheet-icon .x-spreadsheet-icon-img.align-right { 943 | left: 0; 944 | top: -18px; 945 | } 946 | .x-spreadsheet-icon .x-spreadsheet-icon-img.align-top { 947 | left: -18px; 948 | top: -18px; 949 | } 950 | .x-spreadsheet-icon .x-spreadsheet-icon-img.align-middle { 951 | left: -36px; 952 | top: -18px; 953 | } 954 | .x-spreadsheet-icon .x-spreadsheet-icon-img.align-bottom { 955 | left: -54px; 956 | top: -18px; 957 | } 958 | .x-spreadsheet-icon .x-spreadsheet-icon-img.textwrap { 959 | left: -72px; 960 | top: -18px; 961 | } 962 | .x-spreadsheet-icon .x-spreadsheet-icon-img.autofilter { 963 | left: -90px; 964 | top: -18px; 965 | } 966 | .x-spreadsheet-icon .x-spreadsheet-icon-img.formula { 967 | left: -108px; 968 | top: -18px; 969 | } 970 | .x-spreadsheet-icon .x-spreadsheet-icon-img.arrow-down { 971 | left: -126px; 972 | top: -18px; 973 | } 974 | .x-spreadsheet-icon .x-spreadsheet-icon-img.arrow-right { 975 | left: -144px; 976 | top: -18px; 977 | } 978 | .x-spreadsheet-icon .x-spreadsheet-icon-img.link { 979 | left: -162px; 980 | top: -18px; 981 | } 982 | .x-spreadsheet-icon .x-spreadsheet-icon-img.chart { 983 | left: -180px; 984 | top: -18px; 985 | } 986 | .x-spreadsheet-icon .x-spreadsheet-icon-img.freeze { 987 | left: -198px; 988 | top: -18px; 989 | } 990 | .x-spreadsheet-icon .x-spreadsheet-icon-img.ellipsis { 991 | left: -216px; 992 | top: -18px; 993 | } 994 | .x-spreadsheet-icon .x-spreadsheet-icon-img.add { 995 | left: -234px; 996 | top: -18px; 997 | } 998 | .x-spreadsheet-icon .x-spreadsheet-icon-img.border-all { 999 | left: 0; 1000 | top: -36px; 1001 | } 1002 | .x-spreadsheet-icon .x-spreadsheet-icon-img.border-inside { 1003 | left: -18px; 1004 | top: -36px; 1005 | } 1006 | .x-spreadsheet-icon .x-spreadsheet-icon-img.border-horizontal { 1007 | left: -36px; 1008 | top: -36px; 1009 | } 1010 | .x-spreadsheet-icon .x-spreadsheet-icon-img.border-vertical { 1011 | left: -54px; 1012 | top: -36px; 1013 | } 1014 | .x-spreadsheet-icon .x-spreadsheet-icon-img.border-outside { 1015 | left: -72px; 1016 | top: -36px; 1017 | } 1018 | .x-spreadsheet-icon .x-spreadsheet-icon-img.border-left { 1019 | left: -90px; 1020 | top: -36px; 1021 | } 1022 | .x-spreadsheet-icon .x-spreadsheet-icon-img.border-top { 1023 | left: -108px; 1024 | top: -36px; 1025 | } 1026 | .x-spreadsheet-icon .x-spreadsheet-icon-img.border-right { 1027 | left: -126px; 1028 | top: -36px; 1029 | } 1030 | .x-spreadsheet-icon .x-spreadsheet-icon-img.border-bottom { 1031 | left: -144px; 1032 | top: -36px; 1033 | } 1034 | .x-spreadsheet-icon .x-spreadsheet-icon-img.border-none { 1035 | left: -162px; 1036 | top: -36px; 1037 | } 1038 | .x-spreadsheet-icon .x-spreadsheet-icon-img.line-color { 1039 | left: -180px; 1040 | top: -36px; 1041 | } 1042 | .x-spreadsheet-icon .x-spreadsheet-icon-img.line-type { 1043 | left: -198px; 1044 | top: -36px; 1045 | } 1046 | .x-spreadsheet-icon .x-spreadsheet-icon-img.close { 1047 | left: -234px; 1048 | top: -36px; 1049 | } 1050 | .x-spreadsheet-icon .x-spreadsheet-icon-img.chevron-down { 1051 | left: 0; 1052 | top: -54px; 1053 | } 1054 | .x-spreadsheet-icon .x-spreadsheet-icon-img.chevron-up { 1055 | left: -18px; 1056 | top: -54px; 1057 | } 1058 | .x-spreadsheet-icon .x-spreadsheet-icon-img.chevron-left { 1059 | left: -36px; 1060 | top: -54px; 1061 | } 1062 | .x-spreadsheet-icon .x-spreadsheet-icon-img.chevron-right { 1063 | left: -54px; 1064 | top: -54px; 1065 | } 1066 | /*# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["node_modules/x-data-spreadsheet/src/index.less"],
  "sourcesContent": ["body {\n  margin: 0;\n}\n.x-spreadsheet {\n  font-size: 13px;\n  line-height: normal;\n  user-select: none;\n  -moz-user-select: none;\n  font-family: 'Lato', 'Source Sans Pro', Roboto, Helvetica, Arial, sans-serif;\n  box-sizing: content-box;\n  background: #fff;\n  -webkit-font-smoothing: antialiased;\n}\n.x-spreadsheet textarea {\n  font: 400 13px Arial, 'Lato', 'Source Sans Pro', Roboto, Helvetica, sans-serif;\n}\n.x-spreadsheet-sheet {\n  position: relative;\n  overflow: hidden;\n}\n.x-spreadsheet-table {\n  vertical-align: bottom;\n}\n.x-spreadsheet-tooltip {\n  font-family: inherit;\n  position: absolute;\n  padding: 5px 10px;\n  color: #fff;\n  border-radius: 1px;\n  background: #000000;\n  font-size: 12px;\n  z-index: 201;\n}\n.x-spreadsheet-tooltip:before {\n  pointer-events: none;\n  position: absolute;\n  left: calc(50% - 4px);\n  top: -4px;\n  content: \"\";\n  width: 8px;\n  height: 8px;\n  background: inherit;\n  -webkit-transform: rotate(45deg);\n  transform: rotate(45deg);\n  z-index: 1;\n  box-shadow: 1px 1px 3px -1px rgba(0, 0, 0, 0.3);\n}\n.x-spreadsheet-color-palette {\n  padding: 5px;\n}\n.x-spreadsheet-color-palette table {\n  margin: 0;\n  padding: 0;\n  border-collapse: separate;\n  border-spacing: 2;\n  background: #fff;\n}\n.x-spreadsheet-color-palette table td {\n  margin: 0;\n  cursor: pointer;\n  border: 1px solid transparent;\n}\n.x-spreadsheet-color-palette table td:hover {\n  border-color: #ddd;\n}\n.x-spreadsheet-color-palette table td .x-spreadsheet-color-palette-cell {\n  width: 16px;\n  height: 16px;\n}\n.x-spreadsheet-border-palette {\n  padding: 6px;\n}\n.x-spreadsheet-border-palette table {\n  margin: 0;\n  padding: 0;\n  border-collapse: separate;\n  border-spacing: 0;\n  background: #fff;\n  table-layout: fixed;\n}\n.x-spreadsheet-border-palette table td {\n  margin: 0;\n}\n.x-spreadsheet-border-palette .x-spreadsheet-border-palette-left {\n  border-right: 1px solid #eee;\n  padding-right: 6px;\n}\n.x-spreadsheet-border-palette .x-spreadsheet-border-palette-left .x-spreadsheet-border-palette-cell {\n  width: 30px;\n  height: 30px;\n  cursor: pointer;\n  text-align: center;\n}\n.x-spreadsheet-border-palette .x-spreadsheet-border-palette-left .x-spreadsheet-border-palette-cell .x-spreadsheet-icon-img {\n  opacity: 0.8;\n}\n.x-spreadsheet-border-palette .x-spreadsheet-border-palette-left .x-spreadsheet-border-palette-cell:hover {\n  background-color: #eee;\n}\n.x-spreadsheet-border-palette .x-spreadsheet-border-palette-right {\n  padding-left: 6px;\n}\n.x-spreadsheet-border-palette .x-spreadsheet-border-palette-right .x-spreadsheet-toolbar-btn {\n  margin-top: 0;\n  margin-bottom: 3px;\n}\n.x-spreadsheet-border-palette .x-spreadsheet-border-palette-right .x-spreadsheet-line-type {\n  position: relative;\n  left: 0;\n  top: -3px;\n}\n.x-spreadsheet-dropdown {\n  position: relative;\n}\n.x-spreadsheet-dropdown .x-spreadsheet-dropdown-content {\n  position: absolute;\n  z-index: 200;\n  background: #fff;\n  box-shadow: 1px 2px 5px 2px rgba(51, 51, 51, 0.15);\n}\n.x-spreadsheet-dropdown.bottom-left .x-spreadsheet-dropdown-content {\n  top: calc(100% + 5px);\n  left: 0;\n}\n.x-spreadsheet-dropdown.bottom-right .x-spreadsheet-dropdown-content {\n  top: calc(100% + 5px);\n  right: 0;\n}\n.x-spreadsheet-dropdown.top-left .x-spreadsheet-dropdown-content {\n  bottom: calc(100% + 5px);\n  left: 0;\n}\n.x-spreadsheet-dropdown.top-right .x-spreadsheet-dropdown-content {\n  bottom: calc(100% + 5px);\n  right: 0;\n}\n.x-spreadsheet-dropdown .x-spreadsheet-dropdown-title {\n  padding: 0 5px;\n  display: inline-block;\n}\n.x-spreadsheet-dropdown .x-spreadsheet-dropdown-header .x-spreadsheet-icon.arrow-left {\n  margin-left: 4px;\n}\n.x-spreadsheet-dropdown .x-spreadsheet-dropdown-header .x-spreadsheet-icon.arrow-right {\n  width: 10px;\n  margin-right: 4px;\n}\n.x-spreadsheet-dropdown .x-spreadsheet-dropdown-header .x-spreadsheet-icon.arrow-right .arrow-down {\n  left: -130px;\n}\n/* resizer **/\n.x-spreadsheet-resizer {\n  position: absolute;\n  z-index: 11;\n}\n.x-spreadsheet-resizer .x-spreadsheet-resizer-hover {\n  background-color: rgba(75, 137, 255, 0.25);\n}\n.x-spreadsheet-resizer .x-spreadsheet-resizer-line {\n  position: absolute;\n}\n.x-spreadsheet-resizer.horizontal {\n  cursor: row-resize;\n}\n.x-spreadsheet-resizer.horizontal .x-spreadsheet-resizer-line {\n  border-bottom: 2px dashed #4b89ff;\n  left: 0;\n  bottom: 0;\n}\n.x-spreadsheet-resizer.vertical {\n  cursor: col-resize;\n}\n.x-spreadsheet-resizer.vertical .x-spreadsheet-resizer-line {\n  border-right: 2px dashed #4b89ff;\n  top: 0;\n  right: 0;\n}\n/* scrollbar */\n.x-spreadsheet-scrollbar {\n  position: absolute;\n  bottom: 0;\n  right: 0;\n  background-color: #f4f5f8;\n  opacity: 0.9;\n  z-index: 12;\n}\n.x-spreadsheet-scrollbar.horizontal {\n  right: 15px;\n  overflow-x: scroll;\n  overflow-y: hidden;\n}\n.x-spreadsheet-scrollbar.horizontal > div {\n  height: 1px;\n  background: #ddd;\n}\n.x-spreadsheet-scrollbar.vertical {\n  bottom: 15px;\n  overflow-x: hidden;\n  overflow-y: scroll;\n}\n.x-spreadsheet-scrollbar.vertical > div {\n  width: 1px;\n  background: #ddd;\n}\n/* @{css-prefix}-overlayer */\n.x-spreadsheet-overlayer {\n  position: absolute;\n  left: 0;\n  top: 0;\n  z-index: 10;\n}\n.x-spreadsheet-overlayer .x-spreadsheet-overlayer-content {\n  position: absolute;\n  overflow: hidden;\n  pointer-events: none;\n  width: 100%;\n  height: 100%;\n}\n.x-spreadsheet-editor,\n.x-spreadsheet-selector {\n  box-sizing: content-box;\n  position: absolute;\n  overflow: hidden;\n  pointer-events: none;\n  top: 0;\n  left: 0;\n  width: 100%;\n  height: 100%;\n}\n/* @{css-prefix}-selector */\n.x-spreadsheet-selector .hide-input {\n  position: absolute;\n  z-index: 0;\n}\n.x-spreadsheet-selector .hide-input input {\n  padding: 0;\n  width: 0;\n  border: none!important;\n}\n.x-spreadsheet-selector .x-spreadsheet-selector-area {\n  position: absolute;\n  border: 2px solid #4b89ff;\n  background: rgba(75, 137, 255, 0.1);\n  z-index: 5;\n}\n.x-spreadsheet-selector .x-spreadsheet-selector-clipboard,\n.x-spreadsheet-selector .x-spreadsheet-selector-autofill {\n  position: absolute;\n  background: transparent;\n  z-index: 100;\n}\n.x-spreadsheet-selector .x-spreadsheet-selector-clipboard {\n  border: 2px dashed #4b89ff;\n}\n.x-spreadsheet-selector .x-spreadsheet-selector-autofill {\n  border: 1px dashed rgba(0, 0, 0, 0.45);\n}\n.x-spreadsheet-selector .x-spreadsheet-selector-corner {\n  pointer-events: auto;\n  position: absolute;\n  cursor: crosshair;\n  font-size: 0;\n  height: 5px;\n  width: 5px;\n  right: -5px;\n  bottom: -5px;\n  border: 2px solid #ffffff;\n  background: #4b89ff;\n}\n.x-spreadsheet-editor {\n  z-index: 20;\n}\n.x-spreadsheet-editor .x-spreadsheet-editor-area {\n  position: absolute;\n  text-align: left;\n  border: 2px solid #4b89ff;\n  line-height: 0;\n  z-index: 100;\n  pointer-events: auto;\n}\n.x-spreadsheet-editor .x-spreadsheet-editor-area textarea {\n  box-sizing: content-box;\n  border: none;\n  padding: 0 3px;\n  outline: none;\n  resize: none;\n  text-align: start;\n  overflow-y: hidden;\n  font: 400 13px Arial, 'Lato', 'Source Sans Pro', Roboto, Helvetica, sans-serif;\n  color: inherit;\n  white-space: normal;\n  word-wrap: break-word;\n  line-height: 22px;\n  margin: 0;\n}\n.x-spreadsheet-editor .x-spreadsheet-editor-area .textline {\n  overflow: hidden;\n  visibility: hidden;\n  position: fixed;\n  top: 0;\n  left: 0;\n}\n.x-spreadsheet-item {\n  user-select: none;\n  background: 0;\n  border: 1px solid transparent;\n  outline: none;\n  height: 26px;\n  color: rgba(0, 0, 0, 0.9);\n  line-height: 26px;\n  list-style: none;\n  padding: 2px 10px;\n  cursor: default;\n  text-align: left;\n  overflow: hidden;\n}\n.x-spreadsheet-item.disabled {\n  pointer-events: none;\n  opacity: 0.5;\n}\n.x-spreadsheet-item:hover,\n.x-spreadsheet-item.active {\n  background: rgba(0, 0, 0, 0.05);\n}\n.x-spreadsheet-item.divider {\n  height: 0;\n  padding: 0;\n  margin: 5px 0;\n  border: none;\n  border-bottom: 1px solid rgba(0, 0, 0, 0.1);\n}\n.x-spreadsheet-item .label {\n  float: right;\n  opacity: 0.65;\n  font-size: 1em;\n}\n.x-spreadsheet-item.state,\n.x-spreadsheet-header.state {\n  padding-left: 35px!important;\n  position: relative;\n}\n.x-spreadsheet-item.state:before,\n.x-spreadsheet-header.state:before {\n  content: '';\n  position: absolute;\n  width: 10px;\n  height: 10px;\n  left: 12px;\n  top: calc(50% - 5px);\n  background: rgba(0, 0, 0, 0.08);\n  border-radius: 2px;\n}\n.x-spreadsheet-item.state.checked:before,\n.x-spreadsheet-header.state.checked:before {\n  background: #4b89ff;\n}\n.x-spreadsheet-checkbox {\n  position: relative;\n  display: inline-block;\n  backface-visibility: hidden;\n  outline: 0;\n  vertical-align: baseline;\n  font-style: normal;\n  font-size: 1rem;\n  line-height: 1em;\n}\n.x-spreadsheet-checkbox > input {\n  position: absolute;\n  top: 0;\n  left: 0;\n  opacity: 0!important;\n  outline: 0;\n  z-index: -1;\n}\n.x-spreadsheet-suggest,\n.x-spreadsheet-contextmenu,\n.x-spreadsheet-sort-filter {\n  position: absolute;\n  box-shadow: 1px 2px 5px 2px rgba(51, 51, 51, 0.15);\n  background: #fff;\n  z-index: 100;\n  width: 260px;\n  pointer-events: auto;\n  overflow: auto;\n}\n.x-spreadsheet-suggest {\n  width: 200px;\n}\n.x-spreadsheet-filter {\n  border: 1px solid #e9e9e9;\n  font-size: 12px;\n  margin: 10px;\n}\n.x-spreadsheet-filter .x-spreadsheet-header {\n  padding: 0.5em 0.75em;\n  background: #f8f8f9;\n  border-bottom: 1px solid #e9e9e9;\n  border-left: 1px solid transparent;\n}\n.x-spreadsheet-filter .x-spreadsheet-body {\n  height: 200px;\n  overflow-y: auto;\n}\n.x-spreadsheet-filter .x-spreadsheet-body .x-spreadsheet-item {\n  height: 20px;\n  line-height: 20px;\n}\n.x-spreadsheet-sort-filter .x-spreadsheet-buttons {\n  margin: 10px;\n}\n.x-spreadsheet-toolbar,\n.x-spreadsheet-bottombar {\n  height: 40px;\n  padding: 0 30px;\n  text-align: left;\n  background: #f5f6f7;\n  display: flex;\n}\n.x-spreadsheet-bottombar {\n  position: relative;\n  border-top: 1px solid #e0e2e4;\n}\n.x-spreadsheet-bottombar .x-spreadsheet-menu > li {\n  line-height: 40px;\n  height: 40px;\n  padding-top: 0;\n  padding-bottom: 0;\n  vertical-align: middle;\n  border-right: 1px solid #e8eaed;\n}\n.x-spreadsheet-menu {\n  list-style: none;\n  margin: 0;\n  padding: 0;\n  user-select: none;\n}\n.x-spreadsheet-menu > li {\n  float: left;\n  line-height: 1.25em;\n  padding: 0.785em 1em;\n  margin: 0;\n  vertical-align: middle;\n  text-align: left;\n  font-weight: 400;\n  color: #80868b;\n  white-space: nowrap;\n  cursor: pointer;\n  transition: all 0.3s;\n  font-weight: bold;\n}\n.x-spreadsheet-menu > li.active {\n  background-color: #fff;\n  color: rgba(0, 0, 0, 0.65);\n}\n.x-spreadsheet-menu > li .x-spreadsheet-icon {\n  margin: 0 6px;\n}\n.x-spreadsheet-menu > li .x-spreadsheet-icon .x-spreadsheet-icon-img:hover {\n  opacity: 0.85;\n}\n.x-spreadsheet-menu > li .x-spreadsheet-dropdown {\n  display: inline-block;\n}\n.x-spreadsheet-toolbar {\n  border-bottom: 1px solid #e0e2e4;\n}\n.x-spreadsheet-toolbar .x-spreadsheet-toolbar-btns {\n  display: inline-flex;\n}\n.x-spreadsheet-toolbar .x-spreadsheet-toolbar-more {\n  padding: 0 6px 6px;\n  text-align: left;\n}\n.x-spreadsheet-toolbar .x-spreadsheet-toolbar-more .x-spreadsheet-toolbar-divider {\n  margin-top: 0;\n}\n.x-spreadsheet-toolbar .x-spreadsheet-toolbar-btn {\n  flex: 0 0 auto;\n  display: inline-block;\n  border: 1px solid transparent;\n  height: 26px;\n  line-height: 26px;\n  min-width: 26px;\n  margin: 6px 1px 0;\n  padding: 0;\n  text-align: center;\n  border-radius: 2px;\n}\n.x-spreadsheet-toolbar .x-spreadsheet-toolbar-btn.disabled {\n  pointer-events: none;\n  opacity: 0.5;\n}\n.x-spreadsheet-toolbar .x-spreadsheet-toolbar-btn:hover,\n.x-spreadsheet-toolbar .x-spreadsheet-toolbar-btn.active {\n  background: rgba(0, 0, 0, 0.08);\n}\n.x-spreadsheet-toolbar-divider {\n  display: inline-block;\n  border-right: 1px solid #e0e2e4;\n  width: 0;\n  vertical-align: middle;\n  height: 18px;\n  margin: 12px 3px 0;\n}\n.x-spreadsheet-print {\n  position: absolute;\n  left: 0;\n  top: 0;\n  z-index: 100;\n  width: 100%;\n  height: 100%;\n  display: flex;\n  flex-direction: column;\n}\n.x-spreadsheet-print-bar {\n  background: #424242;\n  height: 60px;\n  line-height: 60px;\n  padding: 0 30px;\n}\n.x-spreadsheet-print-bar .-title {\n  color: #fff;\n  font-weight: bold;\n  font-size: 1.2em;\n  float: left;\n}\n.x-spreadsheet-print-bar .-right {\n  float: right;\n  margin-top: 12px;\n}\n.x-spreadsheet-print-content {\n  display: flex;\n  flex: auto;\n  flex-direction: row;\n  background: #d0d0d0;\n  height: calc(100% - 60px);\n}\n.x-spreadsheet-print-content .-sider {\n  flex: 0 0 300px;\n  width: 300px;\n  border-left: 2px solid #ccc;\n  background: #fff;\n}\n.x-spreadsheet-print-content .-content {\n  flex: auto;\n  overflow-x: auto;\n  overflow-y: scroll;\n  height: 100%;\n}\n.x-spreadsheet-canvas-card-wraper {\n  margin: 40px 20px;\n}\n.x-spreadsheet-canvas-card {\n  background: #fff;\n  margin: auto;\n  page-break-before: auto;\n  page-break-after: always;\n  box-shadow: 0 8px 10px 1px rgba(0, 0, 0, 0.14), 0 3px 14px 3px rgba(0, 0, 0, 0.12), 0 4px 5px 0 rgba(0, 0, 0, 0.2);\n}\n.x-spreadsheet-calendar {\n  color: rgba(0, 0, 0, 0.65);\n  background: #ffffff;\n  user-select: none;\n}\n.x-spreadsheet-calendar .calendar-header {\n  font-weight: 700;\n  line-height: 30px;\n  text-align: center;\n  width: 100%;\n  float: left;\n  background: #f9fafb;\n}\n.x-spreadsheet-calendar .calendar-header .calendar-header-left {\n  padding-left: 5px;\n  float: left;\n}\n.x-spreadsheet-calendar .calendar-header .calendar-header-right {\n  float: right;\n}\n.x-spreadsheet-calendar .calendar-header .calendar-header-right a {\n  padding: 3px 0;\n  margin-right: 2px;\n  border-radius: 2px;\n}\n.x-spreadsheet-calendar .calendar-header .calendar-header-right a:hover {\n  background: rgba(0, 0, 0, 0.08);\n}\n.x-spreadsheet-calendar .calendar-body {\n  border-collapse: collapse;\n  border-spacing: 0;\n}\n.x-spreadsheet-calendar .calendar-body th,\n.x-spreadsheet-calendar .calendar-body td {\n  width: 100%/7;\n  min-width: 32px;\n  text-align: center;\n  font-weight: 700;\n  line-height: 30px;\n  padding: 0;\n}\n.x-spreadsheet-calendar .calendar-body td > .cell:hover {\n  background: #ecf6fd;\n}\n.x-spreadsheet-calendar .calendar-body td > .cell.active,\n.x-spreadsheet-calendar .calendar-body td > .cell.active:hover {\n  background: #ecf6fd;\n  color: #2185D0;\n}\n.x-spreadsheet-calendar .calendar-body td > .cell.disabled {\n  pointer-events: none;\n  opacity: 0.5;\n}\n.x-spreadsheet-datepicker {\n  box-shadow: 2px 2px 5px rgba(0, 0, 0, 0.2);\n  position: absolute;\n  left: 0;\n  top: calc(100% + 5px);\n  z-index: 10;\n  width: auto;\n}\n.x-spreadsheet-buttons {\n  display: flex;\n  justify-content: flex-end;\n}\n.x-spreadsheet-buttons .x-spreadsheet-button {\n  margin-left: 8px;\n}\n.x-spreadsheet-button {\n  display: inline-block;\n  border-radius: 3px;\n  line-height: 1em;\n  min-height: 1em;\n  white-space: nowrap;\n  text-align: center;\n  cursor: pointer;\n  font-size: 1em;\n  font-weight: 700;\n  padding: 0.75em 1em;\n  color: rgba(0, 0, 0, 0.6);\n  background: #E0E1E2;\n  text-decoration: none;\n  font-family: \"Lato\", \"proxima-nova\", \"Helvetica Neue\", Arial, sans-serif;\n  outline: none;\n  vertical-align: baseline;\n  zoom: 1;\n  user-select: none;\n  transition: all 0.1s linear;\n}\n.x-spreadsheet-button.active,\n.x-spreadsheet-button:hover {\n  background-color: #C0C1C2;\n  color: rgba(0, 0, 0, 0.8);\n}\n.x-spreadsheet-button.primary {\n  color: #fff;\n  background-color: #2185D0;\n}\n.x-spreadsheet-button.primary:hover,\n.x-spreadsheet-button.primary.active {\n  color: #fff;\n  background-color: #1678c2;\n}\n.x-spreadsheet-form-input {\n  font-size: 1em;\n  position: relative;\n  font-weight: 400;\n  display: inline-flex;\n  color: rgba(0, 0, 0, 0.87);\n}\n.x-spreadsheet-form-input input {\n  z-index: 1;\n  margin: 0;\n  max-width: 100%;\n  flex: 1 0 auto;\n  outline: 0;\n  -webkit-tap-highlight-color: rgba(255, 255, 255, 0);\n  text-align: left;\n  line-height: 30px;\n  height: 30px;\n  padding: 0 8px;\n  background: #fff;\n  border: 1px solid #e9e9e9;\n  border-radius: 3px;\n  transition: box-shadow 0.1s ease, border-color 0.1s ease;\n  box-shadow: inset 0 1px 2px hsla(0, 0%, 4%, 0.06);\n}\n.x-spreadsheet-form-input input:focus {\n  border-color: #4b89ff;\n  box-shadow: inset 0 1px 2px rgba(75, 137, 255, 0.2);\n}\n.x-spreadsheet-form-select {\n  position: relative;\n  display: inline-block;\n  background: #fff;\n  border: 1px solid #e9e9e9;\n  border-radius: 2px;\n  cursor: pointer;\n  color: rgba(0, 0, 0, 0.87);\n  user-select: none;\n  box-shadow: inset 0 1px 2px hsla(0, 0%, 4%, 0.06);\n}\n.x-spreadsheet-form-select .input-text {\n  text-overflow: ellipsis;\n  white-space: nowrap;\n  min-width: 60px;\n  width: auto;\n  height: 30px;\n  line-height: 30px;\n  padding: 0 8px;\n}\n.x-spreadsheet-form-fields {\n  display: flex;\n  flex-direction: row;\n  flex-wrap: wrap;\n}\n.x-spreadsheet-form-fields .x-spreadsheet-form-field {\n  flex: 0 1 auto;\n}\n.x-spreadsheet-form-fields .x-spreadsheet-form-field .label {\n  display: inline-block;\n  margin: 0 10px 0 0;\n}\n.x-spreadsheet-form-field {\n  display: block;\n  vertical-align: middle;\n  margin-left: 10px;\n  margin-bottom: 10px;\n}\n.x-spreadsheet-form-field:first-child {\n  margin-left: 0;\n}\n.x-spreadsheet-form-field.error .x-spreadsheet-form-select,\n.x-spreadsheet-form-field.error input {\n  border-color: #f04134;\n}\n.x-spreadsheet-form-field .tip {\n  color: #f04134;\n  font-size: 0.9em;\n}\n.x-spreadsheet-dimmer {\n  display: none;\n  position: absolute;\n  top: 0 !important;\n  left: 0 !important;\n  width: 100%;\n  height: 100%;\n  text-align: center;\n  vertical-align: middle;\n  background-color: rgba(0, 0, 0, 0.6);\n  opacity: 0;\n  -webkit-animation-fill-mode: both;\n  animation-fill-mode: both;\n  -webkit-animation-duration: 0.5s;\n  animation-duration: 0.5s;\n  transition: background-color 0.5s linear;\n  user-select: none;\n  z-index: 1000;\n}\n.x-spreadsheet-dimmer.active {\n  display: block;\n  opacity: 1;\n}\nform fieldset {\n  border: none;\n}\nform fieldset label {\n  display: block;\n  margin-bottom: 0.5em;\n  font-size: 1em;\n  color: #666;\n}\nform fieldset select {\n  font-size: 1.1em;\n  width: 100%;\n  background-color: #fff;\n  border: none;\n  border-bottom: 2px solid #ddd;\n  padding: 0.5em 0.85em;\n  border-radius: 2px;\n}\n.x-spreadsheet-modal,\n.x-spreadsheet-toast {\n  font-size: 13px;\n  position: fixed;\n  z-index: 1001;\n  text-align: left;\n  line-height: 1.25em;\n  min-width: 360px;\n  color: rgba(0, 0, 0, 0.87);\n  font-family: 'Lato', 'Source Sans Pro', Roboto, Helvetica, Arial, sans-serif;\n  border-radius: 4px;\n  border: 1px solid rgba(0, 0, 0, 0.1);\n  background-color: #fff;\n  background-clip: padding-box;\n  box-shadow: rgba(0, 0, 0, 0.2) 0px 2px 8px;\n}\n.x-spreadsheet-toast {\n  background-color: rgba(255, 255, 255, 0.85);\n}\n.x-spreadsheet-modal-header,\n.x-spreadsheet-toast-header {\n  font-weight: 600;\n  background-clip: padding-box;\n  background-color: rgba(255, 255, 255, 0.85);\n  border-bottom: 1px solid rgba(0, 0, 0, 0.05);\n  border-radius: 4px 4px 0 0;\n}\n.x-spreadsheet-modal-header .x-spreadsheet-icon,\n.x-spreadsheet-toast-header .x-spreadsheet-icon {\n  position: absolute;\n  right: 0.8em;\n  top: 0.65em;\n  border-radius: 18px;\n}\n.x-spreadsheet-modal-header .x-spreadsheet-icon:hover,\n.x-spreadsheet-toast-header .x-spreadsheet-icon:hover {\n  opacity: 1;\n  background: rgba(0, 0, 0, 0.08);\n}\n.x-spreadsheet-toast-header {\n  color: #F2711C;\n}\n.x-spreadsheet-modal-header {\n  border-bottom: 1px solid #e0e2e4;\n  background: rgba(0, 0, 0, 0.08);\n  font-size: 1.0785em;\n}\n.x-spreadsheet-modal-header,\n.x-spreadsheet-modal-content,\n.x-spreadsheet-toast-header,\n.x-spreadsheet-toast-content {\n  padding: 0.75em 1em;\n}\n@media screen and (min-width: 320px) and (max-width: 480px) {\n  .x-spreadsheet-toolbar {\n    display: none;\n  }\n}\n.x-spreadsheet-icon {\n  width: 18px;\n  height: 18px;\n  margin: 1px 1px 2px 1px;\n  text-align: center;\n  vertical-align: middle;\n  user-select: none;\n  overflow: hidden;\n  position: relative;\n  display: inline-block;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img {\n  background-image: url('../assets/sprite.svg');\n  position: absolute;\n  width: 262px;\n  height: 444px;\n  opacity: 0.56;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.undo {\n  left: 0;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.redo {\n  left: -18px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.print {\n  left: -36px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.paintformat {\n  left: -54px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.clearformat {\n  left: -72px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.font-bold {\n  left: -90px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.font-italic {\n  left: -108px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.underline {\n  left: -126px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.strike {\n  left: -144px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.color {\n  left: -162px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.bgcolor {\n  left: -180px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.merge {\n  left: -198px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.align-left {\n  left: -216px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.align-center {\n  left: -234px;\n  top: 0;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.align-right {\n  left: 0;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.align-top {\n  left: -18px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.align-middle {\n  left: -36px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.align-bottom {\n  left: -54px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.textwrap {\n  left: -72px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.autofilter {\n  left: -90px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.formula {\n  left: -108px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.arrow-down {\n  left: -126px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.arrow-right {\n  left: -144px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.link {\n  left: -162px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.chart {\n  left: -180px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.freeze {\n  left: -198px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.ellipsis {\n  left: -216px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.add {\n  left: -234px;\n  top: -18px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.border-all {\n  left: 0;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.border-inside {\n  left: -18px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.border-horizontal {\n  left: -36px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.border-vertical {\n  left: -54px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.border-outside {\n  left: -72px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.border-left {\n  left: -90px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.border-top {\n  left: -108px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.border-right {\n  left: -126px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.border-bottom {\n  left: -144px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.border-none {\n  left: -162px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.line-color {\n  left: -180px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.line-type {\n  left: -198px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.close {\n  left: -234px;\n  top: -36px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.chevron-down {\n  left: 0;\n  top: -54px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.chevron-up {\n  left: -18px;\n  top: -54px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.chevron-left {\n  left: -36px;\n  top: -54px;\n}\n.x-spreadsheet-icon .x-spreadsheet-icon-img.chevron-right {\n  left: -54px;\n  top: -54px;\n}\n"],
  "mappings": ";AAAA;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAGF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAGF;AACE;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAGF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AAAA;AAEF;AAAA;AAEE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAGF;AACE;AACA;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AAAA;AAEE;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AAAA;AAEE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AAAA;AAEE;AACA;AAAA;AAEF;AAAA;AAEE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AAAA;AAEE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AAAA;AAAA;AAGE;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AAAA;AAEE;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AAAA;AAEE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AAAA;AAEE;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AAAA;AAEE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AAAA;AAEE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AAAA;AAEE;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AAAA;AAEE;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AAAA;AAEE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AAAA;AAEE;AACA;AACA;AACA;AACA;AAAA;AAEF;AAAA;AAEE;AACA;AACA;AACA;AAAA;AAEF;AAAA;AAEE;AACA;AAAA;AAEF;AACE;AAAA;AAEF;AACE;AACA;AACA;AAAA;AAEF;AAAA;AAAA;AAAA;AAIE;AAAA;AAEF;AACE;AACE;AAAA;AAAA;AAGJ;AACE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AACA;AACA;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;AAEF;AACE;AACA;AAAA;",
  "names": []
}
 */ 1067 | -------------------------------------------------------------------------------- /manifest-beta.json: -------------------------------------------------------------------------------- 1 | { 2 | "id": "obsidian-workbooks", 3 | "name": "Workbooks", 4 | "version": "1.0.0", 5 | "minAppVersion": "0.15.0", 6 | "description": "Work with Spreadsheets inside your notes", 7 | "author": "Gabriele Cannata", 8 | "authorUrl": "https://github.com/Canna71", 9 | "isDesktopOnly": true 10 | } 11 | -------------------------------------------------------------------------------- /manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "id": "workbooks", 3 | "name": "Workbooks", 4 | "version": "1.0.1", 5 | "minAppVersion": "0.15.0", 6 | "description": "Work with Spreadsheets inside your notes", 7 | "author": "Gabriele Cannata", 8 | "authorUrl": "https://github.com/Canna71", 9 | "isDesktopOnly": true 10 | } 11 | -------------------------------------------------------------------------------- /officeTheme.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 45 | 46 | 47 | 48 | 49 | 50 | 51 | 52 | 53 | 54 | 55 | 56 | 57 | 58 | 59 | 60 | 61 | 62 | 63 | 64 | 65 | 66 | 67 | 68 | 69 | 70 | 71 | 72 | 73 | 74 | 75 | 76 | 77 | 78 | 79 | 80 | 81 | 82 | 83 | 84 | 85 | 86 | 87 | 88 | 89 | 90 | 91 | 92 | 93 | 94 | 95 | 96 | 97 | 98 | 99 | 100 | 101 | 102 | 103 | 104 | 105 | 106 | 107 | 108 | 109 | 110 | 111 | 112 | 113 | 114 | 115 | 116 | 117 | 118 | 119 | 120 | 121 | 122 | 123 | 124 | 125 | 126 | 127 | 128 | 129 | 130 | 131 | 132 | 133 | 134 | 135 | 136 | 137 | 138 | 139 | 140 | 141 | 142 | 143 | 144 | 145 | 146 | 147 | 148 | 149 | 150 | 151 | 152 | 153 | 154 | 155 | 156 | 157 | 158 | 159 | 160 | 161 | 162 | 163 | 164 | 165 | 166 | 167 | 168 | 169 | 170 | 171 | 172 | 173 | 174 | 175 | 176 | 177 | 178 | 179 | 180 | 181 | 182 | 183 | 184 | 185 | 186 | 187 | 188 | 189 | 190 | 191 | 192 | 193 | 194 | 195 | 196 | 197 | 198 | 199 | 200 | 201 | 202 | 203 | 204 | 205 | 206 | 207 | 208 | 209 | 210 | 211 | 212 | 213 | 214 | 215 | 216 | 217 | 218 | 219 | 220 | 221 | 222 | 223 | 224 | 225 | 226 | 227 | 228 | 229 | 230 | 231 | 232 | 233 | 234 | 235 | 236 | 237 | 238 | 239 | 240 | 241 | 242 | 243 | 244 | 245 | 246 | 247 | 248 | 249 | 250 | 251 | 252 | 253 | 254 | 255 | 256 | 257 | 258 | 259 | 260 | 261 | 262 | 263 | 264 | 265 | 266 | 267 | 268 | 269 | 270 | 271 | 272 | 273 | 274 | 275 | 276 | 277 | 278 | 279 | 280 | 281 | 282 | 283 | 284 | 285 | 286 | 287 | 288 | 289 | 290 | 291 | 292 | 293 | 294 | 295 | 296 | 297 | 298 | 299 | 300 | 301 | 302 | 303 | 304 | 305 | 306 | 307 | 308 | 309 | 310 | 311 | 312 | 313 | 314 | 315 | 316 | 317 | 318 | 319 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "obsidian-workbooks", 3 | "version": "1.0.1", 4 | "description": "Work with Spreadsheets inside your notes", 5 | "main": "main.js", 6 | "scripts": { 7 | "dev": "node esbuild.config.mjs", 8 | "build": "tsc -noEmit -skipLibCheck && node esbuild.config.mjs production", 9 | "version": "node version-bump.mjs && git add manifest.json versions.json", 10 | "scss": "ode-sass -w styles.scss styles.css" 11 | }, 12 | "keywords": [], 13 | "author": "", 14 | "license": "MIT", 15 | "devDependencies": { 16 | "@codemirror/language": "^6.2.1", 17 | "@types/node": "^16.11.6", 18 | "@types/react": "^18.0.17", 19 | "@types/react-dom": "^18.0.6", 20 | "@typescript-eslint/eslint-plugin": "5.29.0", 21 | "@typescript-eslint/parser": "5.29.0", 22 | "builtin-modules": "3.3.0", 23 | "esbuild": "0.14.47", 24 | "esbuild-plugin-svgr": "^1.0.1", 25 | "esbuild-sass-plugin": "^2.3.2", 26 | "esbuild-plugin-alias": "^0.2.1", 27 | "esbuild-plugin-less": "^1.2.1", 28 | "obsidian": "latest", 29 | "tslib": "2.4.0", 30 | "typescript": "4.7.4" 31 | }, 32 | "dependencies": { 33 | "exceljs": "^4.3.0", 34 | "react": "^18.2.0", 35 | "react-dom": "^18.2.0", 36 | "xlsx": "https://cdn.sheetjs.com/xlsx-0.20.0/xlsx-0.20.0.tgz" 37 | } 38 | } 39 | -------------------------------------------------------------------------------- /spreadsheetData.json: -------------------------------------------------------------------------------- 1 | [ 2 | { 3 | "name": "Sheet1", 4 | "freeze": "A1", 5 | "styles": [ 6 | { 7 | "bgcolor": "#fff2cd" 8 | }, 9 | { 10 | "bgcolor": "#d9e2f3" 11 | } 12 | ], 13 | "merges": [], 14 | "rows": { 15 | "0": { 16 | "cells": { 17 | "0": { 18 | "text": "Item" 19 | }, 20 | "1": { 21 | "text": "Cost" 22 | }, 23 | "2": { 24 | "text": "Qty" 25 | }, 26 | "3": { 27 | "text": "Price" 28 | } 29 | } 30 | }, 31 | "1": { 32 | "cells": { 33 | "0": { 34 | "text": "Laser", 35 | "style": 0 36 | }, 37 | "1": { 38 | "text": "100", 39 | "style": 0 40 | }, 41 | "2": { 42 | "text": "2", 43 | "style": 0 44 | }, 45 | "3": { 46 | "text": "=B2*C2", 47 | "style": 0 48 | } 49 | } 50 | }, 51 | "2": { 52 | "cells": { 53 | "0": { 54 | "text": "Gun", 55 | "style": 0 56 | }, 57 | "1": { 58 | "text": "123", 59 | "style": 0 60 | }, 61 | "2": { 62 | "text": "3", 63 | "style": 0 64 | }, 65 | "3": { 66 | "text": "=B3*C3", 67 | "style": 0 68 | } 69 | } 70 | }, 71 | "3": { 72 | "cells": { 73 | "0": { 74 | "text": "Some", 75 | "style": 0 76 | }, 77 | "1": { 78 | "text": "234", 79 | "style": 0 80 | }, 81 | "2": { 82 | "text": "1", 83 | "style": 0 84 | }, 85 | "3": { 86 | "text": "=B4*C4", 87 | "style": 0 88 | } 89 | } 90 | }, 91 | "4": { 92 | "cells": { 93 | "0": { 94 | "style": 0 95 | }, 96 | "1": { 97 | "style": 0 98 | }, 99 | "2": { 100 | "text": "3", 101 | "style": 0 102 | }, 103 | "3": { 104 | "text": "=B5*C5", 105 | "style": 0 106 | } 107 | } 108 | }, 109 | "5": { 110 | "cells": { 111 | "0": { 112 | "style": 0 113 | }, 114 | "1": { 115 | "style": 0 116 | }, 117 | "2": { 118 | "style": 0 119 | }, 120 | "3": { 121 | "text": "=B6*C6", 122 | "style": 0 123 | } 124 | } 125 | }, 126 | "6": { 127 | "cells": { 128 | "0": { 129 | "style": 0 130 | }, 131 | "1": { 132 | "style": 0 133 | }, 134 | "2": { 135 | "style": 0 136 | }, 137 | "3": { 138 | "text": "=B7*C7", 139 | "style": 0 140 | } 141 | } 142 | }, 143 | "7": { 144 | "cells": { 145 | "0": { 146 | "style": 0 147 | }, 148 | "1": { 149 | "style": 0 150 | }, 151 | "2": { 152 | "style": 0 153 | }, 154 | "3": { 155 | "text": "=SUM(D2:D7)", 156 | "style": 0 157 | } 158 | } 159 | }, 160 | "len": 100 161 | }, 162 | "cols": { 163 | "len": 26 164 | }, 165 | "validations": [], 166 | "autofilter": {} 167 | }, 168 | { 169 | "name": "sheet3", 170 | "freeze": "A1", 171 | "styles": [], 172 | "merges": [], 173 | "rows": { 174 | "0": { 175 | "cells": { 176 | "0": { 177 | "text": "12" 178 | }, 179 | "1": { 180 | "text": "23" 181 | } 182 | } 183 | }, 184 | "len": 100 185 | }, 186 | "cols": { 187 | "len": 26 188 | }, 189 | "validations": [], 190 | "autofilter": {} 191 | } 192 | ] 193 | -------------------------------------------------------------------------------- /src/Globals.d.ts: -------------------------------------------------------------------------------- 1 | declare module "*.module.css"; 2 | declare module "*.module.scss"; 3 | declare module "*.svg"; 4 | -------------------------------------------------------------------------------- /src/SettingTab.ts: -------------------------------------------------------------------------------- 1 | import SheetjsPlugin from "src/main"; 2 | import { App, PluginSettingTab, Setting } from "obsidian"; 3 | 4 | export class SheetjsSettingsTab extends PluginSettingTab { 5 | plugin: SheetjsPlugin; 6 | 7 | constructor(app: App, plugin: SheetjsPlugin) { 8 | super(app, plugin); 9 | this.plugin = plugin; 10 | } 11 | 12 | display(): void { 13 | const { containerEl } = this; 14 | 15 | containerEl.empty(); 16 | 17 | 18 | this.createToggle( 19 | containerEl, 20 | "Enable saving to file", 21 | "Enabling saving to external files (.xlsx, .xls, ,.csv)", 22 | "enableSaveToFile" 23 | ); 24 | 25 | this.createToggle( 26 | containerEl, 27 | "Auto save", 28 | "Saves automatically", 29 | "autoSave" 30 | ); 31 | } 32 | 33 | private createToggle( 34 | containerEl: HTMLElement, 35 | name: string, 36 | desc: string, 37 | prop: string 38 | ) { 39 | new Setting(containerEl) 40 | .setName(name) 41 | .setDesc(desc) 42 | .addToggle((bool) => 43 | bool 44 | .setValue((this.plugin.settings as any)[prop] as boolean) 45 | .onChange(async (value) => { 46 | (this.plugin.settings as any)[prop] = value; 47 | await this.plugin.saveSettings(); 48 | this.display(); 49 | }) 50 | ); 51 | } 52 | } 53 | -------------------------------------------------------------------------------- /src/Settings.ts: -------------------------------------------------------------------------------- 1 | 2 | export interface SheetsSettings { 3 | enableSaveToFile: boolean; 4 | autoSave: boolean; 5 | } 6 | 7 | export const DEFAULT_SETTINGS: SheetsSettings = { 8 | enableSaveToFile: false, 9 | autoSave: true 10 | } 11 | 12 | -------------------------------------------------------------------------------- /src/Views/SheetView.ts: -------------------------------------------------------------------------------- 1 | /* eslint-disable @typescript-eslint/no-explicit-any */ 2 | // import * as React from "react"; 3 | // import { createRoot } from "react-dom/client"; 4 | // https://github.com/exceljs/exceljs#reading-xlsx 5 | import { 6 | MarkdownPostProcessorContext, 7 | parseYaml 8 | } from "obsidian"; 9 | import { SheetsSettings } from "src/Settings"; 10 | // import "x-data-spreadsheet/dist/xspreadsheet.css"; 11 | // import * as fs from "fs/promises" 12 | // import * as path from "path" 13 | 14 | import * as XLSX from "xlsx"; 15 | import * as ExcelJS from "exceljs"; 16 | import { stox } from "../utils/xlsxpread"; 17 | import { toSpreadsheet } from "src/utils/excelConverter"; 18 | import { 19 | createSpreadSheet, 20 | 21 | saveDataIntoBlock, 22 | 23 | saveToFile, 24 | } from "./spreadSheetWrapper"; 25 | import { Readable } from "stream"; 26 | import { moment } from "obsidian"; 27 | 28 | import saveIcon from "./save.svg"; 29 | 30 | const DEFAULT_OPTIONS = { 31 | height: 540, 32 | width: "auto", 33 | rows: 100, 34 | cols: 26, 35 | fontSize: 10, 36 | cellHeight: 25, 37 | cellWidth: 100, 38 | }; 39 | 40 | const MINHEIGHT = 400; 41 | 42 | // const saveIcon = 43 | // "data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBzdGFuZGFsb25lPSJubyI/PjwhRE9DVFlQRSBzdmcgUFVCTElDICItLy9XM0MvL0RURCBTVkcgMS4xLy9FTiIgImh0dHA6Ly93d3cudzMub3JnL0dyYXBoaWNzL1NWRy8xLjEvRFREL3N2ZzExLmR0ZCI+PHN2ZyB0PSIxNTc3MTc3MDkyOTg4IiBjbGFzcz0iaWNvbiIgdmlld0JveD0iMCAwIDEwMjQgMTAyNCIgdmVyc2lvbj0iMS4xIiB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHAtaWQ9IjI2NzgiIHdpZHRoPSIxOCIgaGVpZ2h0PSIxOCIgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiPjxkZWZzPjxzdHlsZSB0eXBlPSJ0ZXh0L2NzcyI+PC9zdHlsZT48L2RlZnM+PHBhdGggZD0iTTIxMy4zMzMzMzMgMTI4aDU5Ny4zMzMzMzRhODUuMzMzMzMzIDg1LjMzMzMzMyAwIDAgMSA4NS4zMzMzMzMgODUuMzMzMzMzdjU5Ny4zMzMzMzRhODUuMzMzMzMzIDg1LjMzMzMzMyAwIDAgMS04NS4zMzMzMzMgODUuMzMzMzMzSDIxMy4zMzMzMzNhODUuMzMzMzMzIDg1LjMzMzMzMyAwIDAgMS04NS4zMzMzMzMtODUuMzMzMzMzVjIxMy4zMzMzMzNhODUuMzMzMzMzIDg1LjMzMzMzMyAwIDAgMSA4NS4zMzMzMzMtODUuMzMzMzMzeiBtMzY2LjkzMzMzNCAxMjhoMzQuMTMzMzMzYTI1LjYgMjUuNiAwIDAgMSAyNS42IDI1LjZ2MTE5LjQ2NjY2N2EyNS42IDI1LjYgMCAwIDEtMjUuNiAyNS42aC0zNC4xMzMzMzNhMjUuNiAyNS42IDAgMCAxLTI1LjYtMjUuNlYyODEuNmEyNS42IDI1LjYgMCAwIDEgMjUuNi0yNS42ek0yMTMuMzMzMzMzIDIxMy4zMzMzMzN2NTk3LjMzMzMzNGg1OTcuMzMzMzM0VjIxMy4zMzMzMzNIMjEzLjMzMzMzM3ogbTEyOCAwdjI1NmgzNDEuMzMzMzM0VjIxMy4zMzMzMzNoODUuMzMzMzMzdjI5OC42NjY2NjdhNDIuNjY2NjY3IDQyLjY2NjY2NyAwIDAgMS00Mi42NjY2NjcgNDIuNjY2NjY3SDI5OC42NjY2NjdhNDIuNjY2NjY3IDQyLjY2NjY2NyAwIDAgMS00Mi42NjY2NjctNDIuNjY2NjY3VjIxMy4zMzMzMzNoODUuMzMzMzMzek0yNTYgMjEzLjMzMzMzM2g4NS4zMzMzMzMtODUuMzMzMzMzeiBtNDI2LjY2NjY2NyAwaDg1LjMzMzMzMy04NS4zMzMzMzN6IG0wIDU5Ny4zMzMzMzR2LTEyOEgzNDEuMzMzMzMzdjEyOEgyNTZ2LTE3MC42NjY2NjdhNDIuNjY2NjY3IDQyLjY2NjY2NyAwIDAgMSA0Mi42NjY2NjctNDIuNjY2NjY3aDQyNi42NjY2NjZhNDIuNjY2NjY3IDQyLjY2NjY2NyAwIDAgMSA0Mi42NjY2NjcgNDIuNjY2NjY3djE3MC42NjY2NjdoLTg1LjMzMzMzM3ogbTg1LjMzMzMzMyAwaC04NS4zMzMzMzMgODUuMzMzMzMzek0zNDEuMzMzMzMzIDgxMC42NjY2NjdIMjU2aDg1LjMzMzMzM3oiIHAtaWQ9IjI2NzkiIGZpbGw9IiMyYzJjMmMiPjwvcGF0aD48L3N2Zz4="; 44 | 45 | export function processCodeBlock( 46 | source: string, 47 | el: HTMLElement, 48 | settings: SheetsSettings, 49 | ctx: MarkdownPostProcessorContext 50 | ) { 51 | // eslint-disable-next-line @typescript-eslint/no-unused-vars 52 | 53 | // const containerHeight = Math.clamp((ctx as any).containerEl.offsetHeight, 200, 800); 54 | // TODO: check this actually exists 55 | // let bgColor = "#ffffff"; 56 | // let fgColor = "#000"; //"#a0a0a0"; 57 | // const cel = document.getElementsByClassName("view-content")[0]; 58 | // if (cel) { 59 | // const styles = getComputedStyle(cel); 60 | // bgColor = bgColor || styles.getPropertyValue("background"); 61 | // fgColor = fgColor || styles.getPropertyValue("color"); 62 | // } 63 | 64 | if ((ctx as any).spreadsheet) return; 65 | 66 | const options = { ...DEFAULT_OPTIONS, enableSave: settings.enableSaveToFile, autoSave: settings.autoSave, ...parseYaml(source) }; 67 | 68 | options.height = Math.max(options.height, MINHEIGHT); 69 | 70 | const { 71 | filename, 72 | height, 73 | width, 74 | rows, 75 | cols, 76 | fontSize, 77 | cellHeight, 78 | cellWidth, 79 | } = options; 80 | const containerWidth = () => 81 | width === "auto" ? (ctx as any).containerEl.offsetWidth || 1024 : width; 82 | 83 | const container = el; //.createDiv(); 84 | container.style.width = containerWidth() + "px"; 85 | container.tabIndex = -1; 86 | if (container.parentElement) { 87 | container.parentElement.style.overflow = "hidden"; 88 | } 89 | 90 | // @ts-ignore 91 | const view = app.workspace.getActiveFileView(); 92 | const mode = view?.getMode() === "source" ? "edit" : "read"; 93 | 94 | const spreadsheet_options: any = { 95 | mode, // edit | read 96 | showToolbar: true, 97 | showGrid: true, 98 | showContextmenu: true, 99 | showValidation: false, 100 | view: { 101 | height: () => height, 102 | width: () => { 103 | const w = containerWidth(); 104 | return w; 105 | }, 106 | }, 107 | row: { 108 | len: rows, 109 | height: cellHeight, 110 | }, 111 | col: { 112 | len: cols, 113 | width: cellWidth, 114 | }, 115 | style: { 116 | // bgcolor: "#fff", 117 | align: "left", 118 | valign: "middle", 119 | textwrap: false, 120 | strike: false, 121 | underline: false, 122 | // color: "#000", 123 | font: { 124 | // name: font, 125 | size: fontSize, 126 | bold: false, 127 | italic: false, 128 | } as any, 129 | }, 130 | formats: [ 131 | { 132 | key: 'date', 133 | numfmt: moment.localeData().longDateFormat('L').toLowerCase() , 134 | label: moment().format("L"), 135 | title: 'Short Date' 136 | }, 137 | { 138 | key: 'longdate', 139 | numfmt: moment.localeData().longDateFormat('LL').toLowerCase(), 140 | label: moment().format("LL"), 141 | title: 'Long Date' 142 | }, 143 | ] 144 | // onKeyDown: (evt) => { 145 | // } 146 | }; 147 | 148 | if (!filename || options.enableSave) { 149 | const el = document.createElement("div"); 150 | el.innerHTML = saveIcon; 151 | spreadsheet_options.extendToolbar = { 152 | left: [ 153 | { 154 | tip: "Save", 155 | el: el.firstChild, 156 | shortcut: "Ctrl+S", 157 | onClick: (s: any, d: any) => { 158 | if (!filename) saveDataIntoBlock(s, d, ctx); 159 | else { 160 | saveToFile((ctx as any).spreadsheet, filename); 161 | } 162 | }, 163 | }, 164 | ], 165 | }; 166 | } 167 | 168 | // setTimeout(() => { 169 | if (filename !== undefined) { 170 | (async () => { 171 | let data = undefined; 172 | try { 173 | const fileContent = await app.vault.adapter.readBinary( 174 | filename 175 | ); 176 | 177 | data = await parseFileContent(filename, fileContent); 178 | } catch (e) { 179 | console.warn(e); 180 | } 181 | 182 | (ctx as any).spreadsheet = createSpreadSheet( 183 | container, 184 | spreadsheet_options, 185 | { ...options, data: data }, 186 | ctx 187 | ); 188 | // .loadData(); 189 | })(); 190 | } else { 191 | 192 | let wait = 0; 193 | if(!(ctx as any).containerEl.offsetWidth) { 194 | wait = 500; // hack for first opening 195 | } 196 | setTimeout(()=>{ 197 | (ctx as any).spreadsheet = createSpreadSheet( 198 | container, 199 | spreadsheet_options, 200 | { ...options }, 201 | ctx 202 | ); 203 | },wait); 204 | 205 | } 206 | // }, 0); 207 | 208 | // (ctx as any).spreadsheet = s; 209 | 210 | // see https://docs.sheetjs.com/docs/demos/grid/xs 211 | // https://docs.sheetjs.com/xspreadsheet/ 212 | // https://github.com/myliang/x-spreadsheet 213 | // https://forum.obsidian.md/t/saving-changes-in-codeblock-post-processor/47393 214 | // https://codesandbox.io/s/x-spreadsheet-react-3v1bw?file=/src/Spreadsheet.js:527-774 215 | // https://github.com/wolf-table/table 216 | } 217 | 218 | async function parseFileContent(filename: string, fileContent: ArrayBuffer) { 219 | const ext = filename.slice(filename.lastIndexOf(".")).toLowerCase(); 220 | if (ext === ".xlsx" || ext === ".csv") { 221 | const workbook = new ExcelJS.Workbook(); 222 | 223 | // let excelWorkbook : ExcelJS.Workbook | undefined = undefined; 224 | if(ext === ".csv") { 225 | 226 | await workbook.csv.read(new Readable({ 227 | read() { 228 | this.push(Buffer.from(fileContent)); 229 | this.push(null); 230 | } 231 | })); 232 | 233 | } else { 234 | await workbook.xlsx.load(fileContent); 235 | } 236 | 237 | const data2 = toSpreadsheet(workbook); 238 | return data2; 239 | } else { 240 | const xlsx = XLSX.read(fileContent, { 241 | cellStyles: true, 242 | sheetStubs: true, // > 243 | }); 244 | // eslint-disable-next-line @typescript-eslint/no-unused-vars 245 | const data = stox(xlsx); 246 | return data; 247 | } 248 | } 249 | -------------------------------------------------------------------------------- /src/Views/save.svg: -------------------------------------------------------------------------------- 1 | 3 | 6 | -------------------------------------------------------------------------------- /src/Views/spreadSheetWrapper.ts: -------------------------------------------------------------------------------- 1 | // import { Spreadsheet } from 'x-data-spreadsheet'; 2 | /* eslint-disable @typescript-eslint/no-explicit-any */ 3 | import { MarkdownPostProcessorContext, MarkdownView, Notice, debounce, stringifyYaml } from "obsidian"; 4 | import * as XLSX from "xlsx"; 5 | import { xtos } from "../utils/xlsxpread"; 6 | import { toExcelJS } from "src/utils/excelConverter"; 7 | import { SheetData, SpreadsheetData } from "x-data-spreadsheet"; 8 | // HACK 9 | import Spreadsheet from "x-data-spreadsheet"; 10 | import { getSheetjsSettings } from "src/main"; 11 | // import * as Spreadsheet from "x-data-spreadsheet"; 12 | // const { Spreadsheet } = require("x-data-spreadsheet"); 13 | 14 | 15 | function resolve_book_type(fileName: string): XLSX.BookType { 16 | const _BT: any = { 17 | xls: "biff8", 18 | htm: "html", 19 | slk: "sylk", 20 | socialcalc: "eth", 21 | Sh33tJS: "WTF", 22 | }; 23 | let bookType = "xlsx"; 24 | const ext = fileName.slice(fileName.lastIndexOf(".")).toLowerCase(); 25 | if (ext.match(/^\.[a-z]+$/)) { 26 | bookType = ext.slice(1); 27 | } 28 | bookType = _BT[bookType] || bookType; 29 | return bookType as XLSX.BookType; 30 | } 31 | 32 | // eslint-disable-next-line @typescript-eslint/no-unused-vars 33 | function applyStyles(ssdata: any, wb: XLSX.WorkBook) { 34 | for (const sheet of ssdata) { 35 | const { name, styles, rows } = sheet; 36 | for (const rowId in rows) { 37 | const cells = rows[rowId]["cells"]; 38 | for (const cellId in cells) { 39 | const cell = cells[cellId]; 40 | if (cell.style !== undefined) { 41 | const wbStyle = styleSS2WB(styles[cell.style]); 42 | //TODO: apply to the right WB cell 43 | const cellRef = XLSX.utils.encode_cell({ 44 | r: Number(rowId), 45 | c: Number(cellId), 46 | }); 47 | wb.Sheets[name][cellRef].s = wbStyle; 48 | } 49 | } 50 | } 51 | } 52 | } 53 | 54 | 55 | 56 | 57 | export interface SheetOptions { 58 | filename?: string; 59 | data?: any; 60 | enableSave?: boolean; 61 | autoSave?: boolean; 62 | } 63 | 64 | export function createSpreadSheet( 65 | container: HTMLElement, 66 | spreadsheet_options: any, 67 | options: SheetOptions, 68 | ctx: MarkdownPostProcessorContext 69 | ) { 70 | 71 | // const data: SheetData[] = prepareDataForLoading(options.data as SpreadsheetData) 72 | 73 | const spreadSheet = new Spreadsheet( 74 | container, 75 | spreadsheet_options 76 | ) 77 | // .loadData(options.data || {}); 78 | 79 | const settings = getSheetjsSettings() 80 | 81 | prepareDataForLoading(spreadSheet, options.data as SpreadsheetData) 82 | 83 | 84 | if(options.autoSave) { 85 | spreadSheet.change( 86 | debounce((_data) => { 87 | // save data 88 | if (options.filename && options.enableSave) { 89 | saveToFile(spreadSheet, options.filename); 90 | } else { 91 | // at the moment we avoid since this would cause re-rendering 92 | // saveDataIntoBlock(null,null,ctx) 93 | } 94 | 95 | // XLSX.writeFile(xtos(s.getData(data)) as any, filename); 96 | }, 1000) 97 | ); 98 | 99 | 100 | } 101 | 102 | return spreadSheet; 103 | } 104 | 105 | 106 | export function saveDataIntoBlock( 107 | data: any, 108 | sheet: any, 109 | ctx: MarkdownPostProcessorContext 110 | ) { 111 | const s = (ctx as any).spreadsheet as Spreadsheet; 112 | const dts = prepareDataForSaving( s ); 113 | 114 | const view = app.workspace.getActiveViewOfType(MarkdownView); 115 | if (!view) return; 116 | if (view.getMode() === "source") { 117 | const sec = ctx.getSectionInfo((ctx as any).el as HTMLElement); 118 | if (sec) { 119 | const obj = { data: dts }; 120 | const yaml = stringifyYaml(obj) + "\n"; 121 | view?.editor.replaceRange( 122 | yaml, 123 | { line: sec?.lineStart + 1, ch: 0 }, 124 | { line: sec?.lineEnd, ch: 0 }, 125 | "*" 126 | ); 127 | console.info("Data saved on code block"); 128 | } 129 | } else { 130 | // preview 131 | new Notice("Sheet not saved while in reading mode"); 132 | } 133 | } 134 | 135 | 136 | export async function saveToFile(spreadSheet: Spreadsheet, filename: string) { 137 | 138 | const spreadsheetData = spreadSheet.getData() as any[]; 139 | const bookType = resolve_book_type(filename); 140 | if(bookType === 'xlsx' || bookType === 'csv'){ 141 | const workbook = toExcelJS(spreadsheetData); 142 | if(bookType === 'xlsx'){ 143 | const buffer = await workbook.xlsx.writeBuffer(); 144 | app.vault.adapter.writeBinary(filename, buffer); 145 | } else { 146 | const buffer = await workbook.csv.writeBuffer(); 147 | app.vault.adapter.writeBinary(filename, buffer); 148 | } 149 | 150 | } else { 151 | const wb = xtos(spreadsheetData) as XLSX.WorkBook; 152 | // applyStyles(spreadsheetData, wb); 153 | const bytes = XLSX.write(wb, { 154 | bookType: bookType, 155 | type: "buffer", 156 | compression: true, 157 | bookSST: true, 158 | cellStyles: true, 159 | }); 160 | app.vault.adapter.writeBinary(filename, bytes); 161 | } 162 | 163 | 164 | 165 | 166 | 167 | 168 | // fs.writeFile(filename,bytes); 169 | 170 | } 171 | 172 | function styleSS2WB(ssstyle: any) { 173 | const style: any = { patternType: "solid" }; 174 | if (ssstyle.bgcolor) { 175 | style.bgColor = { 176 | rgb: ssstyle.bgcolor.substring(1), 177 | }; 178 | } 179 | 180 | if (ssstyle.color) { 181 | style.fgColor = { 182 | rgb: ssstyle.color.substring(1), 183 | }; 184 | } 185 | return style; 186 | } 187 | 188 | 189 | export function prepareDataForSaving(spreadSheet: Spreadsheet): SpreadsheetData { 190 | const data = spreadSheet.getData() as SheetData[]; 191 | 192 | // get some info 193 | const selector = (spreadSheet as any).sheet.data.selector; 194 | const sheetName = (spreadSheet as any).sheet.data.name; 195 | 196 | 197 | 198 | for(const sheet of data){ 199 | const actualStyles = []; 200 | const usedStyles = new Map(); 201 | if(sheet.styles !== undefined) { 202 | for(const rowId in sheet.rows) { 203 | const rowNum = Number(rowId) 204 | if(!isNaN(rowNum)) { 205 | const row = sheet.rows[rowNum]; 206 | for(const cellId in row.cells) { 207 | const cellNum = Number(cellId); 208 | const cell = row.cells[cellNum]; 209 | if(cell.style !== undefined){ 210 | if(usedStyles.has(cell.style)){ 211 | cell.style = usedStyles.get(cell.style) 212 | } else { 213 | actualStyles.push(sheet.styles[cell.style]) 214 | const index = actualStyles.length-1; 215 | usedStyles.set(cell.style, index) 216 | cell.style = index; 217 | } 218 | } 219 | } 220 | } 221 | } 222 | } 223 | sheet.styles = actualStyles; 224 | } 225 | 226 | const spreadSheetData : SpreadsheetData = {...data} 227 | 228 | spreadSheetData.state = { 229 | sheetName, 230 | selector 231 | } 232 | 233 | return spreadSheetData; 234 | } 235 | 236 | function prepareDataForLoading(spreadsheet:Spreadsheet, spreadSheetData: SpreadsheetData): Spreadsheet { 237 | if(spreadSheetData === undefined){ 238 | return spreadsheet.loadData({}); 239 | } else { 240 | const sheets = [] 241 | for(const sheetId in spreadSheetData){ 242 | const sheetNum = Number(sheetId) 243 | if(!isNaN(sheetNum)){ 244 | sheets[sheetNum] = spreadSheetData[sheetId] 245 | } 246 | } 247 | spreadsheet.loadData(sheets); 248 | if(spreadSheetData.state?.sheetName){ 249 | // const d = this.datas[index]; 250 | // this.sheet.resetData(d); 251 | const s = (spreadsheet as any); 252 | // const d = s.datas.find(d => d.name === spreadSheetData.state?.sheetName) 253 | const i = s.datas.findIndex((d:any) => d.name === spreadSheetData.state?.sheetName) 254 | 255 | const d = s.datas[i]; 256 | const selector = spreadSheetData.state?.selector 257 | if(selector){ 258 | // d.selector = spreadSheetData.state?.selector; 259 | d.selector.setIndexes(selector.ri, selector.ci); 260 | d.selector.range.sci = selector.range.sci; 261 | d.selector.range.sri = selector.range.sri; 262 | d.selector.range.eci = selector.range.eci; 263 | d.selector.range.eri = selector.range.eri; 264 | d.selector.range.h = selector.range.h; 265 | d.selector.range.w = selector.range.w; 266 | 267 | } 268 | if(i>=0){ 269 | // TODO: provide an ad hoc method in x-spreadsheet 270 | s.bottombar.clickSwap2(s.bottombar.items[i]); 271 | } 272 | 273 | } 274 | return spreadsheet; 275 | } 276 | } 277 | -------------------------------------------------------------------------------- /src/main.ts: -------------------------------------------------------------------------------- 1 | import { DEFAULT_SETTINGS, SheetsSettings } from "src/Settings"; 2 | import { addIcon, MarkdownView } from "obsidian"; 3 | 4 | 5 | // import { MathResult } from './Extensions/ResultMarkdownChild'; 6 | /* eslint-disable @typescript-eslint/no-unused-vars */ 7 | import { 8 | App, 9 | finishRenderMath, 10 | loadMathJax, 11 | Modal, 12 | Plugin, 13 | WorkspaceLeaf, 14 | } from "obsidian"; 15 | import { SheetjsSettingsTab } from "src/SettingTab"; 16 | import { processCodeBlock } from "./Views/SheetView"; 17 | 18 | 19 | const sheetSVG = ` 20 | `; 21 | 22 | // Remember to rename these classes and interfaces! 23 | 24 | let gSettings: SheetsSettings; 25 | 26 | export function getSheetjsSettings() { return gSettings; } 27 | export default class SheetjsPlugin extends Plugin { 28 | settings: SheetsSettings; 29 | 30 | async onload() { 31 | await this.loadSettings(); 32 | 33 | 34 | addIcon("sheet",sheetSVG); 35 | 36 | 37 | this.registerCodeBlock(); 38 | 39 | 40 | this.addSettingTab(new SheetjsSettingsTab(this.app, this)); 41 | } 42 | 43 | 44 | 45 | 46 | onunload() { 47 | } 48 | 49 | async loadSettings() { 50 | this.settings = Object.assign( 51 | {}, 52 | DEFAULT_SETTINGS, 53 | await this.loadData() 54 | ); 55 | gSettings = this.settings; 56 | } 57 | 58 | async saveSettings() { 59 | await this.saveData(this.settings); 60 | } 61 | 62 | 63 | 64 | async registerCodeBlock() { 65 | 66 | this.registerMarkdownCodeBlockProcessor( 67 | "sheet", 68 | (source, el, ctx) => { 69 | processCodeBlock(source, el, this.settings, ctx); 70 | } 71 | ); 72 | } 73 | 74 | } 75 | 76 | 77 | -------------------------------------------------------------------------------- /src/utils/excelColors.ts: -------------------------------------------------------------------------------- 1 | // Sample theme color index 2 | // const themeColorIndex = 1; // The theme color index (0-9) 3 | // const tintVal = -0.5; // The tint value (-1 to 1) 4 | 5 | // Sample Office Color Themes 6 | const officeThemeColors = [ 7 | [255, 255, 255], // White 8 | [0, 0, 0], // Black 9 | [238, 236, 225], // Tan 10 | [31, 73, 125], // Dark Blue 11 | [79, 129, 189], // Blue 12 | [192, 80, 77], // Red 13 | [155, 187, 89], // Green 14 | [128, 100, 162], // Purple 15 | [75, 172, 198], // Aqua 16 | [245, 150, 70], // Orange 17 | ]; 18 | 19 | const RGBToHSL = (r:number, g:number, b:number) => { 20 | r /= 255; 21 | g /= 255; 22 | b /= 255; 23 | const l = Math.max(r, g, b); 24 | const s = l - Math.min(r, g, b); 25 | const h = s 26 | ? l === r 27 | ? (g - b) / s 28 | : l === g 29 | ? 2 + (b - r) / s 30 | : 4 + (r - g) / s 31 | : 0; 32 | return [ 33 | 60 * h < 0 ? 60 * h + 360 : 60 * h, 34 | 100 * (s ? (l <= 0.5 ? s / (2 * l - s) : s / (2 - (2 * l - s))) : 0), 35 | (100 * (2 * l - s)) / 2, 36 | ]; 37 | }; 38 | 39 | // TODO: read theme from XML 40 | export function convertThemeColorToRGB( 41 | themeColorIndex: number, 42 | tintVal: number 43 | ) { 44 | const baseColor = officeThemeColors[themeColorIndex]; 45 | const baseHSL = RGBToHSL(baseColor[0],baseColor[1], baseColor[2]) //rgb2hsl(baseColor); 46 | const lumination = baseHSL[2]; 47 | if (tintVal < 0) { 48 | baseHSL[2] = lumination * (1.0 + tintVal); 49 | } else { 50 | baseHSL[2] = lumination * (1.0 - tintVal) + 100 * tintVal; 51 | } 52 | 53 | // return HSLToRGB(baseHSL[0] / 360, baseHSL[1] / 100, baseHSL[2] / 100); 54 | return hsl2rgb(baseHSL[0], baseHSL[1] / 100, baseHSL[2] / 100); 55 | } 56 | 57 | // input: h as an angle in [0,360] and s,l in [0,1] 58 | function hsl2rgb(h:number,s:number,l:number) 59 | { 60 | const a=s*Math.min(l,1-l); 61 | const f= (n:number,k=(n+h/30)%12) => l - a*Math.max(Math.min(k-3,9-k,1),-1); 62 | return [f(0)*255,f(8)*255,f(4)*255].map(Math.round); 63 | } 64 | 65 | export function rgbToHex(r:number, g:number, b:number) { 66 | return r.toString(16) + g.toString(16) + b.toString(16); 67 | } 68 | -------------------------------------------------------------------------------- /src/utils/excelConverter.ts: -------------------------------------------------------------------------------- 1 | /* eslint-disable @typescript-eslint/no-explicit-any */ 2 | import { 3 | BorderStyle, 4 | Borders, 5 | CellFormulaValue, 6 | Column, 7 | Table, 8 | Workbook, 9 | } from "exceljs"; 10 | import { convertThemeColorToRGB, rgbToHex } from "./excelColors"; 11 | import { CellData, CellStyle, RowData, SheetData } from "x-data-spreadsheet"; 12 | // import { SpreadsheetData } from "x-data-spreadsheet"; 13 | 14 | declare module "x-data-spreadsheet" { 15 | interface RowData { 16 | height?: number; 17 | } 18 | } 19 | 20 | type borderDir = "top" | "bottom" | "left" | "right"; 21 | 22 | export function toSpreadsheet(wb: Workbook) { 23 | function mapColor( 24 | oStyle: CellStyle, 25 | border: Partial, 26 | what: borderDir 27 | ) { 28 | if (border[what] && border[what]?.style && oStyle.border) { 29 | oStyle.border[what] = [ 30 | // eslint-disable-next-line @typescript-eslint/no-non-null-assertion 31 | border[what]!.style!.toString(), 32 | // eslint-disable-next-line @typescript-eslint/no-non-null-assertion 33 | "#" + getColor(border[what]!.color), 34 | ]; 35 | } 36 | } 37 | 38 | 39 | 40 | const out = wb.worksheets.map((ws) => { 41 | const ows: SheetData = { 42 | name: ws.name, 43 | rows: {}, 44 | cols: {}, 45 | merges: [], 46 | 47 | // freezes: 48 | }; 49 | 50 | ows.styles = []; 51 | 52 | if (ws.columns) { 53 | for (const col of ws.columns) { 54 | if (col.width && ows.cols && col.number) { 55 | ows.cols[col.number - 1] = { width: width2px(col.width) }; 56 | } 57 | } 58 | } 59 | // TODO: use options 60 | ows.cols!.len = Math.max(25, ws.actualColumnCount); 61 | 62 | ws.eachRow((row, rowNumber) => { 63 | const rowId = row.number - 1; 64 | const rowob: RowData = { 65 | cells: {}, 66 | }; 67 | 68 | if (ows.rows) { 69 | ows.rows[rowId] = rowob; 70 | } 71 | if (row.height) { 72 | // px = pt * ( 72pt / 96 ) 73 | 74 | rowob.height = pt2px(row.height); 75 | } 76 | if (row.hidden) { 77 | rowob.hidden = true; 78 | // TODO: implementit! 79 | } 80 | 81 | row.eachCell((cell, cellNumber) => { 82 | const cellOb: CellData = { 83 | text: "" 84 | }; 85 | 86 | try { 87 | cellOb.text = cell.text 88 | } catch {} 89 | 90 | if (cell.formula) cellOb.text = "=" + cell.formula; 91 | 92 | // style 93 | const oStyle: CellStyle = {}; 94 | if (cell.style.fill) { 95 | if (cell.style.fill.type === "pattern") { 96 | const fgColor = cell.style.fill.fgColor; 97 | if (fgColor) { 98 | const hexColor = getColor(fgColor); 99 | if (hexColor) { 100 | oStyle.bgcolor = "#" + hexColor; 101 | } 102 | } 103 | } 104 | } 105 | 106 | if (cell.style.border) { 107 | const border = cell.style.border; 108 | oStyle.border = {}; 109 | ["top", "bottom", "left", "right"].forEach((what) => { 110 | mapColor(oStyle, border, what as borderDir); 111 | }); 112 | } 113 | if (cell.style.font) { 114 | const font = cell.style.font; 115 | 116 | if (font.bold) { 117 | oStyle.font = oStyle.font || {}; 118 | oStyle.font.bold = true; 119 | } 120 | if (font.color) { 121 | const hexColor = getColor(font.color); 122 | if (hexColor) { 123 | oStyle.color = "#" + hexColor; 124 | } 125 | } 126 | if (font.italic) { 127 | oStyle.font = oStyle.font || {}; 128 | // oStyle.font 129 | oStyle.font.italic = true; 130 | } 131 | if (font.strike) { 132 | oStyle.strike = true; 133 | } 134 | if (font.underline) { 135 | oStyle.underline = true; 136 | } 137 | if (font.name) { 138 | oStyle.font = oStyle.font || {}; 139 | oStyle.font.name = font.name; 140 | } 141 | if (font.family) { 142 | oStyle.font = oStyle.font || {}; 143 | oStyle.font.family = font.family; 144 | } 145 | if (font.size) { 146 | oStyle.font = oStyle.font || {}; 147 | oStyle.font.size = font.size; 148 | } 149 | } 150 | if (cell.style.alignment) { 151 | // 152 | const alig = cell.style.alignment; 153 | if (alig.vertical) { 154 | (oStyle as any).valign = alig.vertical; 155 | } 156 | if (alig.horizontal) { 157 | (oStyle as any).align = alig.horizontal; 158 | } 159 | if (cell.style.alignment.wrapText) { 160 | oStyle.textwrap = true; 161 | } 162 | } 163 | if (cell.style.numFmt) { 164 | const numFmt = cell.style.numFmt; 165 | oStyle.format = numFmt; 166 | } 167 | if (cell.style.protection) { 168 | //TODO: 169 | } 170 | 171 | if (Object.keys(oStyle).length > 0 && ows.styles) { 172 | const j = JSON.stringify(oStyle); 173 | let styleIndex = ows.styles.findIndex( 174 | (s) => JSON.stringify(s) == j 175 | ); 176 | if (styleIndex < 0) { 177 | ows.styles?.push(oStyle); 178 | styleIndex = ows.styles.length - 1; 179 | } 180 | cellOb.style = styleIndex; 181 | } 182 | 183 | if (!cell.isMerged || !cell.model.master) { 184 | rowob.cells[Number(cell.col) - 1] = cellOb; 185 | if (cell.isMerged) { 186 | const merge = (ws as any)._merges[cell.address]; 187 | cellOb.merge = [ 188 | (merge.bottom - merge.top) as number, 189 | (merge.right - merge.left) as number, 190 | ]; 191 | } 192 | } else { 193 | // 194 | } 195 | }); 196 | }); 197 | 198 | // TODO: use options 199 | ows.rows!.len = Math.max(100, ws.actualRowCount); 200 | // merges 201 | const merges = []; 202 | for (const m in (ws as any)._merges) { 203 | const merge = (ws as any)._merges[m]; 204 | const range = merge.range; 205 | merges.push(range); 206 | } 207 | ows.merges = merges; 208 | 209 | if(typeof(ws.autoFilter) === "string"){ 210 | ows.autofilter = { 211 | ref: ws.autoFilter, 212 | filters: [], 213 | }; 214 | } 215 | 216 | return ows; 217 | }); 218 | 219 | return out; 220 | } 221 | function pt2px(pt: number) { 222 | return Math.round(pt * 1.3333333); 223 | } 224 | 225 | function px2pt(px: number) { 226 | return Math.round(px * 0.75); 227 | } 228 | 229 | function width2px(w: number) { 230 | // TODO: get actual character width 231 | // 10 units = 64px 232 | return Math.round(w * 6.4); 233 | } 234 | 235 | function px2width(px: number) { 236 | return Math.round(px * 0.15625); 237 | } 238 | 239 | function getColor(fgColor: any) { 240 | if (fgColor.argb) { 241 | const hex = fgColor.argb.substring(2); 242 | return hex; 243 | } else if (fgColor.theme !== undefined) { 244 | const theme = fgColor?.theme; 245 | const tint = (fgColor as any).tint || 0; 246 | const rgb = convertThemeColorToRGB(theme, tint); 247 | const hex = rgbToHex(rgb[0], rgb[1], rgb[2]); 248 | return hex; 249 | } 250 | } 251 | 252 | function toColor(hex: string) { 253 | return { 254 | argb: "FF" + hex.replace("#", ""), 255 | }; 256 | } 257 | 258 | export function toExcelJS(data: SheetData[]): Workbook { 259 | function mapBorderColor( 260 | borders: Partial, 261 | cellstyle: CellStyle, 262 | what: borderDir 263 | ) { 264 | if (cellstyle.border) { 265 | const brd = cellstyle.border[what]; 266 | if (brd) { 267 | const [style, color] = brd; 268 | borders[what] = { 269 | style: style as BorderStyle, 270 | color: toColor(color), 271 | }; 272 | } 273 | } 274 | } 275 | 276 | const workbook = new Workbook(); 277 | for (const ssheet of data) { 278 | const wsheet = workbook.addWorksheet(ssheet.name); 279 | 280 | if (ssheet.cols !== undefined) { 281 | const colIds = Object.keys(ssheet.cols) 282 | .filter((key) => !isNaN(Number(key))) 283 | .map((key) => Number(key)); 284 | 285 | const maxColid = Math.max(...colIds); 286 | 287 | const wscols: Partial[] = []; 288 | for (let colid = 0; colid <= maxColid; colid++) { 289 | const num = colid + 1; 290 | const col: Partial = { 291 | number: num, 292 | key: num.toString(), 293 | }; 294 | const scol = ssheet.cols[colid]; 295 | 296 | if (scol?.width) { 297 | col.width = px2width(scol.width); 298 | } 299 | wscols.push(col); 300 | } 301 | 302 | wsheet.columns = wscols; 303 | } 304 | 305 | if (ssheet.rows !== undefined) { 306 | for (const rowId in ssheet.rows) { 307 | if (!isNaN(Number(rowId))) { 308 | const rowNum = Number(rowId); 309 | const rowdata = ssheet.rows[rowNum]; 310 | 311 | const row = wsheet.getRow(rowNum + 1); 312 | // cells, height, hidden 313 | if (rowdata.hidden) { 314 | row.hidden = true; 315 | } 316 | if (rowdata.height !== undefined) { 317 | row.height = px2pt(rowdata.height); 318 | } 319 | for (const cellId in rowdata.cells) { 320 | const cellNum = Number(cellId); 321 | const celldata = rowdata.cells[cellNum]; 322 | const cell = row.getCell(cellNum + 1); 323 | 324 | if (celldata.text?.startsWith("=")) { 325 | cell.value = { 326 | formula: celldata.text.substring(1), 327 | } as CellFormulaValue; 328 | } else if (!isNaN(Number(celldata.text))) { 329 | cell.value = Number(celldata.text); 330 | } else if (!isNaN(Date.parse(celldata.text))) { 331 | cell.value = new Date(celldata.text); 332 | } else { 333 | cell.value = celldata.text; 334 | } 335 | 336 | // style 337 | if (celldata.style !== undefined) { 338 | const cellstyle = ssheet.styles![celldata.style]; 339 | if (cellstyle.bgcolor) { 340 | cell.style.fill = { 341 | type: "pattern", 342 | pattern: "solid", 343 | fgColor: toColor(cellstyle.bgcolor), 344 | }; 345 | } 346 | 347 | if (cellstyle.border) { 348 | const borders: Partial = {}; 349 | 350 | ["top", "bottom", "right", "left"].forEach( 351 | (what) => { 352 | mapBorderColor( 353 | borders, 354 | cellstyle, 355 | what as borderDir 356 | ); 357 | } 358 | ); 359 | cell.style.border = borders; 360 | } 361 | 362 | if (cellstyle.font) { 363 | cell.style.font = cell.style.font || {}; 364 | if (cellstyle.font.bold) 365 | cell.style.font.bold = true; 366 | if (cellstyle.font.family !== undefined) 367 | cell.style.font.family = 368 | cellstyle.font.family; 369 | if (cellstyle.font.italic) 370 | cell.style.font.italic = true; 371 | if (cellstyle.font.name) 372 | cell.style.font.name = cellstyle.font.name; 373 | if (cellstyle.font.size !== undefined) 374 | cell.style.font.size = cellstyle.font.size; 375 | } 376 | if (cellstyle.color !== undefined) { 377 | cell.style.font = cell.style.font || {}; 378 | cell.style.font.color = getColor( 379 | cellstyle.color 380 | ); 381 | } 382 | if (cellstyle.strike) { 383 | cell.style.font = cell.style.font || {}; 384 | cell.style.font.strike = true; 385 | } 386 | if (cellstyle.underline) { 387 | cell.style.font = cell.style.font || {}; 388 | cell.style.font.underline = true; 389 | } 390 | if (cellstyle.textwrap) { 391 | cell.style.alignment = 392 | cell.style.alignment || {}; 393 | cell.style.alignment.wrapText = true; 394 | } 395 | 396 | if (cellstyle.align !== undefined) { 397 | cell.style.alignment = 398 | cell.style.alignment || {}; 399 | cell.style.alignment.horizontal = 400 | cellstyle.align; 401 | } 402 | 403 | if (cellstyle.valign !== undefined) { 404 | cell.style.alignment = 405 | cell.style.alignment || {}; 406 | cell.style.alignment.vertical = 407 | cellstyle.valign; 408 | } 409 | 410 | if ( 411 | cellstyle.format !== undefined && 412 | cellstyle.format !== "" 413 | ) { 414 | cell.style.numFmt = cellstyle.format; 415 | // if(cellstyle.format === "percent") { 416 | // cell.style.numFmt = "0.00%" 417 | // } else if (cellstyle.format === "eur") { 418 | // cell.style.numFmt = `#,##0.00 "€"` 419 | // } else if (cellstyle.format === "usd") { 420 | // cell.style.numFmt = `#,##0.00 "$"` 421 | // } else if (cellstyle.format === "date") { 422 | // cell.style.numFmt === "mm-dd-yy" 423 | // } else if (cellstyle.format === "time") { 424 | // cell.style.numFmt === "[$-F400]h:mm:ss AM/PM" 425 | // } 426 | } 427 | } 428 | 429 | // cell.numFmt = "00.00" 430 | } 431 | } 432 | } 433 | } 434 | 435 | const merges = ssheet.merges; 436 | if (merges) { 437 | merges.forEach((merge) => { 438 | wsheet.mergeCells(merge); 439 | }); 440 | } 441 | 442 | if (ssheet.autofilter) { 443 | wsheet.autoFilter = ssheet.autofilter.ref; 444 | // (wsheet as any).autoFilter = ssheet.autofilter; 445 | } 446 | } 447 | 448 | // workbook.creator 449 | // workbook.category 450 | // workbook.company 451 | // workbook.created // Date 452 | // workbook.creator 453 | // workbook.keywords 454 | // workbook.lastModifiedBy 455 | // workbook.manager 456 | // workbook.modified // Date 457 | // workbook.subject 458 | // workbook.title 459 | 460 | return workbook; 461 | } 462 | -------------------------------------------------------------------------------- /src/utils/xlsxpread.js: -------------------------------------------------------------------------------- 1 | /*! xlsxspread.js (C) SheetJS LLC -- https://sheetjs.com/ */ 2 | /* eslint-env browser */ 3 | /*global XLSX */ 4 | /*exported stox, xtos */ 5 | 6 | import * as XLSX from "xlsx" 7 | 8 | /** 9 | * Converts data from SheetJS to x-spreadsheet 10 | * 11 | * @param {Object} wb SheetJS workbook object 12 | * 13 | * @returns {Object[]} An x-spreadsheet data 14 | */ 15 | export function stox(wb) { 16 | var out = []; 17 | wb.SheetNames.forEach(function (name) { 18 | var o = { name: name, rows: {} }; 19 | var ws = wb.Sheets[name]; 20 | if(!ws || !ws["!ref"]) return; 21 | var range = XLSX.utils.decode_range(ws['!ref']); 22 | // sheet_to_json will lost empty row and col at begin as default 23 | range.s = { r: 0, c: 0 }; 24 | var aoa = XLSX.utils.sheet_to_json(ws, { 25 | raw: false, 26 | header: 1, 27 | range: range 28 | }); 29 | 30 | aoa.forEach(function (r, i) { 31 | var cells = {}; 32 | r.forEach(function (c, j) { 33 | cells[j] = { text: c }; 34 | 35 | var cellRef = XLSX.utils.encode_cell({ r: i, c: j }); 36 | 37 | if ( ws[cellRef] != null && ws[cellRef].f != null) { 38 | cells[j].text = "=" + ws[cellRef].f; 39 | } 40 | }); 41 | o.rows[i] = { cells: cells }; 42 | }); 43 | 44 | o.merges = []; 45 | (ws["!merges"]||[]).forEach(function (merge, i) { 46 | //Needed to support merged cells with empty content 47 | if (o.rows[merge.s.r] == null) { 48 | o.rows[merge.s.r] = { cells: {} }; 49 | } 50 | if (o.rows[merge.s.r].cells[merge.s.c] == null) { 51 | o.rows[merge.s.r].cells[merge.s.c] = {}; 52 | } 53 | 54 | o.rows[merge.s.r].cells[merge.s.c].merge = [ 55 | merge.e.r - merge.s.r, 56 | merge.e.c - merge.s.c 57 | ]; 58 | 59 | o.merges[i] = XLSX.utils.encode_range(merge); 60 | }); 61 | 62 | out.push(o); 63 | }); 64 | 65 | return out; 66 | } 67 | 68 | /** 69 | * Converts data from x-spreadsheet to SheetJS 70 | * 71 | * @param {Object[]} sdata An x-spreadsheet data object 72 | * 73 | * @returns {Object} A SheetJS workbook object 74 | */ 75 | export function xtos(sdata) { 76 | var out = XLSX.utils.book_new(); 77 | sdata.forEach(function (xws) { 78 | var ws = {}; 79 | var rowobj = xws.rows; 80 | var minCoord = { r: 0, c: 0 }, maxCoord = { r: 0, c: 0 }; 81 | for (var ri = 0; ri < rowobj.len; ++ri) { 82 | var row = rowobj[ri]; 83 | if (!row) continue; 84 | 85 | Object.keys(row.cells).forEach(function (k) { 86 | var idx = +k; 87 | if (isNaN(idx)) return; 88 | 89 | var lastRef = XLSX.utils.encode_cell({ r: ri, c: idx }); 90 | if (ri > maxCoord.r) maxCoord.r = ri; 91 | if (idx > maxCoord.c) maxCoord.c = idx; 92 | 93 | var cellText = row.cells[k].text, type = "s"; 94 | if (!cellText) { 95 | cellText = ""; 96 | type = "z"; 97 | } else if (!isNaN(Number(cellText))) { 98 | cellText = Number(cellText); 99 | type = "n"; 100 | } else if (cellText.toLowerCase() === "true" || cellText.toLowerCase() === "false") { 101 | cellText = Boolean(cellText); 102 | type = "b"; 103 | } 104 | 105 | ws[lastRef] = { v: cellText, t: type }; 106 | 107 | if (type == "s" && cellText[0] == "=") { 108 | ws[lastRef].f = cellText.slice(1); 109 | } 110 | 111 | if (row.cells[k].merge != null) { 112 | if (ws["!merges"] == null) ws["!merges"] = []; 113 | 114 | ws["!merges"].push({ 115 | s: { r: ri, c: idx }, 116 | e: { 117 | r: ri + row.cells[k].merge[0], 118 | c: idx + row.cells[k].merge[1] 119 | } 120 | }); 121 | } 122 | }); 123 | } 124 | ws["!ref"] = minCoord ? XLSX.utils.encode_range({ 125 | s: minCoord, 126 | e: maxCoord 127 | }) : "A1"; 128 | 129 | XLSX.utils.book_append_sheet(out, ws, xws.name); 130 | }); 131 | 132 | return out; 133 | } -------------------------------------------------------------------------------- /styles.scss: -------------------------------------------------------------------------------- 1 | // @import "x-data-spreadsheet/dist/xspreadsheet.css" 2 | // 3 | @import "node_modules/x-data-spreadsheet/dist/xspreadsheet"; 4 | 5 | // override 6 | 7 | // .x-spreadsheet { 8 | // background: var(--background-primary); 9 | // } 10 | 11 | // .x-spreadsheet { 12 | // --ss-accent-h: var(--accent-h); 13 | // --ss-accent-s: var(--accent-s); 14 | // --ss-accent-l: var(--accent-l); 15 | // } 16 | 17 | .x-spreadsheet { 18 | 19 | --ss-accent-h: var(--accent-h); 20 | --ss-accent-s: var(--accent-s); 21 | --ss-accent-l: var(--accent-l); 22 | --ss-background: var(--background-primary); 23 | --ss-background-secondary: var(--background-secondary); 24 | --ss-background-alt: var(--background-primary-alt); 25 | --ss-background-secondary-alt: var(--background-secondary-alt); 26 | --ss-text-color: var(--text-normal); 27 | --ss-text-color-muted: var(--text-muted); 28 | --ss-border-color: var(--divider-color); 29 | --item-hover: var(--background-modifier-hover); 30 | --toolbar-bg-active: var(--background-modifier-hover); 31 | } 32 | 33 | // div.x-spreadsheet-sheet 34 | // > div.x-spreadsheet-overlayer 35 | // > div 36 | // > div.x-spreadsheet-editor 37 | // > div 38 | // > textarea { 39 | // background-color: #ffffff; 40 | // // background-color: var(--background-primary); 41 | 42 | // color: #000000; 43 | // // color: var(--text-normal); 44 | // box-shadow: inherit; 45 | // } 46 | .x-spreadsheet-editor { 47 | textarea { 48 | border-radius: 0px;; 49 | } 50 | } 51 | // .x-spreadsheet-tooltip { 52 | // color: var(--text-normal); 53 | // background: var(--background-primary); 54 | // } 55 | 56 | // .x-spreadsheet-color-palette table td:hover { 57 | // border-color: var(--background-modifier-border-hover); 58 | // } 59 | 60 | // .x-spreadsheet-border-palette 61 | // .x-spreadsheet-border-palette-left 62 | // .x-spreadsheet-border-palette-cell:hover { 63 | // background-color: var(--background-primary); 64 | // } 65 | 66 | // .x-spreadsheet-resizer .x-spreadsheet-resizer-hover { 67 | // background-color: rgba(75, 137, 255, 0.25); 68 | // } 69 | 70 | // .x-spreadsheet-item { 71 | // background: var(--background-secondary); 72 | 73 | // color: var(--text-muted); 74 | // } 75 | 76 | // .x-spreadsheet-menu > li { 77 | // color: var(--text-muted); 78 | 79 | // &.active { 80 | // background-color: var(--background-primary); 81 | // color: var(--text-normal); 82 | // } 83 | // } 84 | 85 | // .x-spreadsheet-print-bar .-title { 86 | // color: var(--text-normal); 87 | // } 88 | 89 | // .x-spreadsheet-calendar { 90 | // color: var(--text-normal); 91 | // background: var(--background-primary); 92 | 93 | // .calendar-body td > .cell.active { 94 | // background: var(--background-primary); 95 | 96 | // color: var(--text-normal); 97 | // } 98 | 99 | // .calendar-header { 100 | // background: var(--background-secondary); 101 | 102 | // .calendar-header-right a:hover { 103 | // background: rgba(0, 0, 0, 0.08); 104 | // } 105 | // } 106 | 107 | // .calendar-body td > .cell { 108 | // &:hover { 109 | // background: #ecf6fd; 110 | // } 111 | 112 | // &.active, 113 | // &.active:hover { 114 | // background: #ecf6fd; 115 | // color: #2185d0; 116 | // } 117 | // } 118 | // } 119 | 120 | // .x-spreadsheet-datepicker { 121 | // box-shadow: 2px 2px 5px rgba(0, 0, 0, 0.2); 122 | // } 123 | 124 | // .x-spreadsheet-button { 125 | // color: var(--text-muted); 126 | // background: var(--background-secondary); 127 | 128 | // &.active, 129 | // &:hover { 130 | // background-color: var(--background-secondary); 131 | // color: var(--text-normal); 132 | // } 133 | // } 134 | 135 | // .x-spreadsheet-button.primary { 136 | // color: var(--text-normal); 137 | // background-color: var(--background-secondary); 138 | 139 | // &:hover, 140 | // &.active { 141 | // color: var(--text-normal); 142 | // background-color: var(--background-secondary); 143 | // } 144 | // } 145 | 146 | // .x-spreadsheet-form-input { 147 | // color: rgba(0, 0, 0, 0.87); 148 | 149 | // input { 150 | // -webkit-tap-highlight-color: rgba(255, 255, 255, 0); 151 | // background: var(--background-primary); 152 | // border: 1px solid var(--background-modifier-border-hover); 153 | // box-shadow: inset 0 1px 2px hsla(0, 0%, 4%, 0.06); 154 | 155 | // &:focus { 156 | // border-color: #123456; 157 | // box-shadow: inset 0 1px 2px rgba(12, 34, 56, 0.2); 158 | // } 159 | // } 160 | // } 161 | 162 | // .x-spreadsheet-form-select { 163 | // background: #fff; 164 | // border: 1px solid #e9e9e9; 165 | // color: rgba(0, 0, 0, 0.87); 166 | // box-shadow: inset 0 1px 2px hsla(0, 0%, 4%, 0.06); 167 | // } 168 | 169 | // .x-spreadsheet-form-field.error .x-spreadsheet-form-select, 170 | // .x-spreadsheet-form-field.error input { 171 | // border-color: #f04134; 172 | // } 173 | // .x-spreadsheet-form-field .tip { 174 | // color: #f04134; 175 | // } 176 | 177 | // .x-spreadsheet-toolbar, 178 | // .x-spreadsheet-bottombar { 179 | 180 | // background: var(--background-secondary); 181 | 182 | // } 183 | 184 | // .x-spreadsheet-toolbar-btn { 185 | // color: var(--text-muted); 186 | // background: var(--background-secondary); 187 | // } 188 | 189 | div.x-spreadsheet { 190 | overflow: hidden; 191 | } 192 | 193 | .x-spreadsheet-dimmer.active { 194 | display: none; 195 | z-index: -1; 196 | } 197 | 198 | ul.x-spreadsheet-menu { 199 | padding: 0px; 200 | margin: 0px; 201 | } 202 | 203 | div.x-spreadsheet-toolbar div.x-spreadsheet-icon { 204 | cursor: pointer; 205 | img { 206 | cursor: pointer; 207 | } 208 | } 209 | 210 | .x-spreadsheet-border-palette table tr td { 211 | overflow-y: visible; 212 | overflow-x: visible; 213 | } 214 | 215 | 216 | 217 | 218 | .x-spreadsheet-icon-img.arrow-down { 219 | margin: 0; 220 | border: 0; 221 | } 222 | 223 | // TODO: create custom classes for each dropdown item, then sets limits accordingly 224 | // we suppose a min height of 400px for the whole spreadhsheet 225 | 226 | // .x-spreadsheet-dropdown .x-spreadsheet-dropdown-content { 227 | // max-height: 300px; 228 | // overflow-y: auto; 229 | // } 230 | 231 | // .x-spreadsheet-dropdown-content .x-spreadsheet-dropdown-content { 232 | // z-index: 201; 233 | // .x-spreadsheet-dropdown-content { 234 | // z-index: 202; 235 | // } 236 | // } 237 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "baseUrl": ".", 4 | "paths": { 5 | "x-data-spreadsheet": ["node_modules/x-data-spreadsheet"] 6 | }, 7 | "inlineSourceMap": true, 8 | "inlineSources": true, 9 | "module": "ESNext", 10 | "target": "ES6", 11 | "allowJs": true, 12 | "noImplicitAny": true, 13 | "moduleResolution": "node", 14 | "importHelpers": true, 15 | "isolatedModules": true, 16 | "strictNullChecks": true, 17 | "esModuleInterop": false, 18 | // "allowSyntheticDefaultImports": true, 19 | "lib": [ 20 | "DOM", 21 | "ES5", 22 | "ES6", 23 | "ES7" 24 | ], 25 | "jsx": "react" 26 | }, 27 | "include": [ 28 | "**/*.ts","**/*.tsx","**/*.svg" 29 | ], 30 | "exclude": ["node_modules"] 31 | } 32 | -------------------------------------------------------------------------------- /version-bump.mjs: -------------------------------------------------------------------------------- 1 | import { readFileSync, writeFileSync } from "fs"; 2 | 3 | const targetVersion = process.env.npm_package_version; 4 | 5 | // read minAppVersion from manifest.json and bump version to target version 6 | let manifest = JSON.parse(readFileSync("manifest.json", "utf8")); 7 | const { minAppVersion } = manifest; 8 | manifest.version = targetVersion; 9 | writeFileSync("manifest.json", JSON.stringify(manifest, null, "\t")); 10 | 11 | // update versions.json with target version and minAppVersion from manifest.json 12 | let versions = JSON.parse(readFileSync("versions.json", "utf8")); 13 | versions[targetVersion] = minAppVersion; 14 | writeFileSync("versions.json", JSON.stringify(versions, null, "\t")); 15 | -------------------------------------------------------------------------------- /versions.json: -------------------------------------------------------------------------------- 1 | { 2 | "1.0.0": "0.15.0" 3 | } 4 | -------------------------------------------------------------------------------- /wb.json: -------------------------------------------------------------------------------- 1 | { 2 | "opts": { 3 | "Date1904": false, 4 | "CalcPrecision": true, 5 | "RefreshAll": false, 6 | "FullCalc": false, 7 | "CalcMode": 1, 8 | "CalcCount": 100, 9 | "CalcIter": false, 10 | "CalcDelta": 0.001, 11 | "CalcSaveRecalc": true 12 | }, 13 | "SheetNames": [ 14 | "Sheet1", 15 | "sheet3" 16 | ], 17 | "Sheets": { 18 | "Sheet1": { 19 | "!margins": { 20 | "left": 0.75, 21 | "right": 0.75, 22 | "top": 1, 23 | "bottom": 1, 24 | "header": 0.5, 25 | "footer": 0.5 26 | }, 27 | "A1": { 28 | "v": "Item", 29 | "t": "s", 30 | "z": "General", 31 | "w": "Item" 32 | }, 33 | "B1": { 34 | "v": "Cost", 35 | "t": "s", 36 | "z": "General", 37 | "w": "Cost" 38 | }, 39 | "C1": { 40 | "v": "Qty", 41 | "t": "s", 42 | "z": "General", 43 | "w": "Qty" 44 | }, 45 | "D1": { 46 | "v": "Price", 47 | "t": "s", 48 | "z": "General", 49 | "w": "Price" 50 | }, 51 | "A2": { 52 | "v": "Laser", 53 | "t": "s", 54 | "z": "General", 55 | "w": "Laser", 56 | "s": { 57 | "patternType": "solid", 58 | "fgColor": { 59 | "rgb": "CCFFCC" 60 | }, 61 | "bgColor": { 62 | "rgb": "000000" 63 | } 64 | } 65 | }, 66 | "B2": { 67 | "v": 100, 68 | "t": "n", 69 | "z": "General", 70 | "w": "100", 71 | "s": { 72 | "patternType": "solid", 73 | "fgColor": { 74 | "rgb": "CCFFCC" 75 | }, 76 | "bgColor": { 77 | "rgb": "000000" 78 | } 79 | } 80 | }, 81 | "C2": { 82 | "v": 2, 83 | "t": "n", 84 | "z": "General", 85 | "w": "2", 86 | "s": { 87 | "patternType": "solid", 88 | "fgColor": { 89 | "rgb": "CCFFCC" 90 | }, 91 | "bgColor": { 92 | "rgb": "000000" 93 | } 94 | } 95 | }, 96 | "D2": { 97 | "v": "=B2*C2", 98 | "t": "s", 99 | "z": "General", 100 | "w": "=B2*C2", 101 | "s": { 102 | "patternType": "solid", 103 | "fgColor": { 104 | "rgb": "CCFFCC" 105 | }, 106 | "bgColor": { 107 | "rgb": "000000" 108 | } 109 | } 110 | }, 111 | "A3": { 112 | "v": "Gun", 113 | "t": "s", 114 | "z": "General", 115 | "w": "Gun", 116 | "s": { 117 | "patternType": "solid", 118 | "fgColor": { 119 | "rgb": "CCFFCC" 120 | }, 121 | "bgColor": { 122 | "rgb": "000000" 123 | } 124 | } 125 | }, 126 | "B3": { 127 | "v": 123, 128 | "t": "n", 129 | "z": "General", 130 | "w": "123", 131 | "s": { 132 | "patternType": "solid", 133 | "fgColor": { 134 | "rgb": "CCFFCC" 135 | }, 136 | "bgColor": { 137 | "rgb": "000000" 138 | } 139 | } 140 | }, 141 | "C3": { 142 | "v": 3, 143 | "t": "n", 144 | "z": "General", 145 | "w": "3", 146 | "s": { 147 | "patternType": "solid", 148 | "fgColor": { 149 | "rgb": "CCFFCC" 150 | }, 151 | "bgColor": { 152 | "rgb": "000000" 153 | } 154 | } 155 | }, 156 | "D3": { 157 | "v": "=B3*C3", 158 | "t": "s", 159 | "z": "General", 160 | "w": "=B3*C3", 161 | "s": { 162 | "patternType": "solid", 163 | "fgColor": { 164 | "rgb": "CCFFCC" 165 | }, 166 | "bgColor": { 167 | "rgb": "000000" 168 | } 169 | } 170 | }, 171 | "A4": { 172 | "v": "Some", 173 | "t": "s", 174 | "z": "General", 175 | "w": "Some", 176 | "s": { 177 | "patternType": "solid", 178 | "fgColor": { 179 | "rgb": "CCFFCC" 180 | }, 181 | "bgColor": { 182 | "rgb": "000000" 183 | } 184 | } 185 | }, 186 | "B4": { 187 | "v": 234, 188 | "t": "n", 189 | "z": "General", 190 | "w": "234", 191 | "s": { 192 | "patternType": "solid", 193 | "fgColor": { 194 | "rgb": "CCFFCC" 195 | }, 196 | "bgColor": { 197 | "rgb": "000000" 198 | } 199 | } 200 | }, 201 | "C4": { 202 | "v": 1, 203 | "t": "n", 204 | "z": "General", 205 | "w": "1", 206 | "s": { 207 | "patternType": "solid", 208 | "fgColor": { 209 | "rgb": "CCFFCC" 210 | }, 211 | "bgColor": { 212 | "rgb": "000000" 213 | } 214 | } 215 | }, 216 | "D4": { 217 | "v": "=B4*C4", 218 | "t": "s", 219 | "z": "General", 220 | "w": "=B4*C4", 221 | "s": { 222 | "patternType": "solid", 223 | "fgColor": { 224 | "rgb": "CCFFCC" 225 | }, 226 | "bgColor": { 227 | "rgb": "000000" 228 | } 229 | } 230 | }, 231 | "A5": { 232 | "t": "z" 233 | }, 234 | "B5": { 235 | "t": "z" 236 | }, 237 | "C5": { 238 | "v": 3, 239 | "t": "n", 240 | "z": "General", 241 | "w": "3" 242 | }, 243 | "D5": { 244 | "v": "=B5*C5", 245 | "t": "s", 246 | "z": "General", 247 | "w": "=B5*C5" 248 | }, 249 | "A6": { 250 | "t": "z" 251 | }, 252 | "B6": { 253 | "t": "z" 254 | }, 255 | "C6": { 256 | "t": "z" 257 | }, 258 | "D6": { 259 | "v": "=B6*C6", 260 | "t": "s", 261 | "z": "General", 262 | "w": "=B6*C6" 263 | }, 264 | "A7": { 265 | "t": "z" 266 | }, 267 | "B7": { 268 | "t": "z" 269 | }, 270 | "C7": { 271 | "t": "z" 272 | }, 273 | "D7": { 274 | "v": "=B7*C7", 275 | "t": "s", 276 | "z": "General", 277 | "w": "=B7*C7" 278 | }, 279 | "A8": { 280 | "t": "z" 281 | }, 282 | "B8": { 283 | "t": "z" 284 | }, 285 | "C8": { 286 | "t": "z" 287 | }, 288 | "D8": { 289 | "v": "=SUM(D2:D7)", 290 | "t": "s", 291 | "z": "General", 292 | "w": "=SUM(D2:D7)" 293 | }, 294 | "!ref": "A1:D8" 295 | }, 296 | "sheet3": { 297 | "!margins": { 298 | "left": 0.75, 299 | "right": 0.75, 300 | "top": 1, 301 | "bottom": 1, 302 | "header": 0.5, 303 | "footer": 0.5 304 | }, 305 | "A1": { 306 | "v": 12, 307 | "t": "n", 308 | "z": "General", 309 | "w": "12" 310 | }, 311 | "B1": { 312 | "v": 23, 313 | "t": "n", 314 | "z": "General", 315 | "w": "23" 316 | }, 317 | "A2": { 318 | "t": "z", 319 | "s": { 320 | "patternType": "solid", 321 | "fgColor": { 322 | "rgb": "CCFFCC" 323 | }, 324 | "bgColor": { 325 | "rgb": "000000" 326 | } 327 | } 328 | }, 329 | "B2": { 330 | "t": "z", 331 | "s": { 332 | "patternType": "solid", 333 | "fgColor": { 334 | "rgb": "CCFFCC" 335 | }, 336 | "bgColor": { 337 | "rgb": "000000" 338 | } 339 | } 340 | }, 341 | "C2": { 342 | "t": "z", 343 | "s": { 344 | "patternType": "solid", 345 | "fgColor": { 346 | "rgb": "CCFFCC" 347 | }, 348 | "bgColor": { 349 | "rgb": "000000" 350 | } 351 | } 352 | }, 353 | "D2": { 354 | "t": "z", 355 | "s": { 356 | "patternType": "solid", 357 | "fgColor": { 358 | "rgb": "CCFFCC" 359 | }, 360 | "bgColor": { 361 | "rgb": "000000" 362 | } 363 | } 364 | }, 365 | "A3": { 366 | "t": "z", 367 | "s": { 368 | "patternType": "solid", 369 | "fgColor": { 370 | "rgb": "CCFFCC" 371 | }, 372 | "bgColor": { 373 | "rgb": "000000" 374 | } 375 | } 376 | }, 377 | "B3": { 378 | "t": "z", 379 | "s": { 380 | "patternType": "solid", 381 | "fgColor": { 382 | "rgb": "CCFFCC" 383 | }, 384 | "bgColor": { 385 | "rgb": "000000" 386 | } 387 | } 388 | }, 389 | "C3": { 390 | "t": "z", 391 | "s": { 392 | "patternType": "solid", 393 | "fgColor": { 394 | "rgb": "CCFFCC" 395 | }, 396 | "bgColor": { 397 | "rgb": "000000" 398 | } 399 | } 400 | }, 401 | "D3": { 402 | "t": "z", 403 | "s": { 404 | "patternType": "solid", 405 | "fgColor": { 406 | "rgb": "CCFFCC" 407 | }, 408 | "bgColor": { 409 | "rgb": "000000" 410 | } 411 | } 412 | }, 413 | "A4": { 414 | "t": "z", 415 | "s": { 416 | "patternType": "solid", 417 | "fgColor": { 418 | "rgb": "CCFFCC" 419 | }, 420 | "bgColor": { 421 | "rgb": "000000" 422 | } 423 | } 424 | }, 425 | "B4": { 426 | "t": "z", 427 | "s": { 428 | "patternType": "solid", 429 | "fgColor": { 430 | "rgb": "CCFFCC" 431 | }, 432 | "bgColor": { 433 | "rgb": "000000" 434 | } 435 | } 436 | }, 437 | "C4": { 438 | "t": "z", 439 | "s": { 440 | "patternType": "solid", 441 | "fgColor": { 442 | "rgb": "CCFFCC" 443 | }, 444 | "bgColor": { 445 | "rgb": "000000" 446 | } 447 | } 448 | }, 449 | "D4": { 450 | "t": "z", 451 | "s": { 452 | "patternType": "solid", 453 | "fgColor": { 454 | "rgb": "CCFFCC" 455 | }, 456 | "bgColor": { 457 | "rgb": "000000" 458 | } 459 | } 460 | }, 461 | "!ref": "A1:D4" 462 | } 463 | }, 464 | "Preamble": { 465 | "!protect": false 466 | }, 467 | "Strings": [ 468 | { 469 | "t": "Item", 470 | "raw": "Item", 471 | "r": "Item" 472 | }, 473 | { 474 | "t": "Cost", 475 | "raw": "Cost", 476 | "r": "Cost" 477 | }, 478 | { 479 | "t": "Qty", 480 | "raw": "Qty", 481 | "r": "Qty" 482 | }, 483 | { 484 | "t": "Price", 485 | "raw": "Price", 486 | "r": "Price" 487 | }, 488 | { 489 | "t": "Laser", 490 | "raw": "Laser", 491 | "r": "Laser" 492 | }, 493 | { 494 | "t": "=B2*C2", 495 | "raw": "=B2*C2", 496 | "r": "=B2*C2" 497 | }, 498 | { 499 | "t": "Gun", 500 | "raw": "Gun", 501 | "r": "Gun" 502 | }, 503 | { 504 | "t": "=B3*C3", 505 | "raw": "=B3*C3", 506 | "r": "=B3*C3" 507 | }, 508 | { 509 | "t": "Some", 510 | "raw": "Some", 511 | "r": "Some" 512 | }, 513 | { 514 | "t": "=B4*C4", 515 | "raw": "=B4*C4", 516 | "r": "=B4*C4" 517 | }, 518 | { 519 | "t": "=B5*C5", 520 | "raw": "=B5*C5", 521 | "r": "=B5*C5" 522 | }, 523 | { 524 | "t": "=B6*C6", 525 | "raw": "=B6*C6", 526 | "r": "=B6*C6" 527 | }, 528 | { 529 | "t": "=B7*C7", 530 | "raw": "=B7*C7", 531 | "r": "=B7*C7" 532 | }, 533 | { 534 | "t": "=SUM(D2:D7)", 535 | "raw": "=SUM(D2:D7)", 536 | "r": "=SUM(D2:D7)" 537 | } 538 | ], 539 | "SSF": { 540 | "0": "General", 541 | "1": "0", 542 | "2": "0.00", 543 | "3": "#,##0", 544 | "4": "#,##0.00", 545 | "5": "#,##0\\ \"€\";\\-#,##0\\ \"€\"", 546 | "6": "#,##0\\ \"€\";[Red]\\-#,##0\\ \"€\"", 547 | "7": "#,##0.00\\ \"€\";\\-#,##0.00\\ \"€\"", 548 | "8": "#,##0.00\\ \"€\";[Red]\\-#,##0.00\\ \"€\"", 549 | "9": "0%", 550 | "10": "0.00%", 551 | "11": "0.00E+00", 552 | "12": "# ?/?", 553 | "13": "# ??/??", 554 | "14": "m/d/yy", 555 | "15": "d-mmm-yy", 556 | "16": "d-mmm", 557 | "17": "mmm-yy", 558 | "18": "h:mm AM/PM", 559 | "19": "h:mm:ss AM/PM", 560 | "20": "h:mm", 561 | "21": "h:mm:ss", 562 | "22": "m/d/yy h:mm", 563 | "37": "#,##0 ;(#,##0)", 564 | "38": "#,##0 ;[Red](#,##0)", 565 | "39": "#,##0.00;(#,##0.00)", 566 | "40": "#,##0.00;[Red](#,##0.00)", 567 | "41": "_-* #,##0_-;\\-* #,##0_-;_-* \"-\"_-;_-@_-", 568 | "42": "_-* #,##0\\ \"€\"_-;\\-* #,##0\\ \"€\"_-;_-* \"-\"\\ \"€\"_-;_-@_-", 569 | "43": "_-* #,##0.00_-;\\-* #,##0.00_-;_-* \"-\"??_-;_-@_-", 570 | "44": "_-* #,##0.00\\ \"€\"_-;\\-* #,##0.00\\ \"€\"_-;_-* \"-\"??\\ \"€\"_-;_-@_-", 571 | "45": "mm:ss", 572 | "46": "[h]:mm:ss", 573 | "47": "mmss.0", 574 | "48": "##0.0E+0", 575 | "49": "@", 576 | "56": "\"上午/下午 \"hh\"時\"mm\"分\"ss\"秒 \"", 577 | "164": "\"上午/下午 \"hh\"時\"mm\"分\"ss\"秒 \"" 578 | }, 579 | "Themes": { 580 | "themeElements": { 581 | "clrScheme": [ 582 | { 583 | "name": "lt1", 584 | "rgb": "FFFFFF" 585 | }, 586 | { 587 | "name": "dk1", 588 | "rgb": "000000" 589 | }, 590 | { 591 | "name": "lt2", 592 | "rgb": "E7E6E6" 593 | }, 594 | { 595 | "name": "dk2", 596 | "rgb": "44546A" 597 | }, 598 | { 599 | "name": "accent1", 600 | "rgb": "4472C4" 601 | }, 602 | { 603 | "name": "accent2", 604 | "rgb": "ED7D31" 605 | }, 606 | { 607 | "name": "accent3", 608 | "rgb": "A5A5A5" 609 | }, 610 | { 611 | "name": "accent4", 612 | "rgb": "FFC000" 613 | }, 614 | { 615 | "name": "accent5", 616 | "rgb": "5B9BD5" 617 | }, 618 | { 619 | "name": "accent6", 620 | "rgb": "70AD47" 621 | }, 622 | { 623 | "name": "hlink", 624 | "rgb": "0563C1" 625 | }, 626 | { 627 | "name": "folHlink", 628 | "rgb": "954F72" 629 | } 630 | ] 631 | }, 632 | "raw": "\r\n" 633 | }, 634 | "Metadata": { 635 | "Country": [ 636 | "US", 637 | "US" 638 | ] 639 | }, 640 | "Workbook": { 641 | "Sheets": [ 642 | { 643 | "Hidden": 0, 644 | "name": "Sheet1", 645 | "CodeName": "Sheet1" 646 | }, 647 | { 648 | "Hidden": 0, 649 | "name": "sheet3", 650 | "CodeName": "sheet3" 651 | } 652 | ], 653 | "WBProps": { 654 | "date1904": false 655 | }, 656 | "Views": [ 657 | {} 658 | ] 659 | }, 660 | "Custprops": { 661 | "SystemIdentifier": 66573, 662 | "CodePage": 10000, 663 | "AppVersion": "16.0000", 664 | "ScaleCrop": false, 665 | "LinksUpToDate": false, 666 | "SharedDoc": false, 667 | "HyperlinksChanged": false, 668 | "FMTID": "02d5cdd59c2e1b10939708002b2cf9ae", 669 | "LastAuthor": "Gabriele Cannata", 670 | "Application": "Microsoft Macintosh Excel", 671 | "ModifiedDate": "2023-08-04T11:22:18Z", 672 | "DocSecurity": 0, 673 | "Worksheets": 2, 674 | "SheetNames": [ 675 | "Sheet1", 676 | "sheet3" 677 | ] 678 | }, 679 | "Props": { 680 | "SystemIdentifier": 66573, 681 | "CodePage": 10000, 682 | "AppVersion": "16.0000", 683 | "ScaleCrop": false, 684 | "LinksUpToDate": false, 685 | "SharedDoc": false, 686 | "HyperlinksChanged": false, 687 | "FMTID": "02d5cdd59c2e1b10939708002b2cf9ae", 688 | "LastAuthor": "Gabriele Cannata", 689 | "Application": "Microsoft Macintosh Excel", 690 | "ModifiedDate": "2023-08-04T11:22:18Z", 691 | "DocSecurity": 0, 692 | "Worksheets": 2, 693 | "SheetNames": [ 694 | "Sheet1", 695 | "sheet3" 696 | ] 697 | } 698 | } 699 | --------------------------------------------------------------------------------