diff --git a/blocks/browse/da-actionbar/da-actionbar.js b/blocks/browse/da-actionbar/da-actionbar.js index 63d45039..01c8923f 100644 --- a/blocks/browse/da-actionbar/da-actionbar.js +++ b/blocks/browse/da-actionbar/da-actionbar.js @@ -8,7 +8,8 @@ const STYLE = await getStyle(import.meta.url); export default class DaActionBar extends LitElement { static properties = { items: { attribute: false }, - _canPaste: { state: true }, + permissions: { attribute: false }, + _isCopying: { state: true }, _isDeleting: { state: true }, _isMoving: { state: true }, currentPath: { type: String }, @@ -29,7 +30,7 @@ export default class DaActionBar extends LitElement { if (props.has('items')) { // Reset state when items go empty if (this.items.length === 0) { - this._canPaste = false; + this._isCopying = false; this._isMoving = false; this._isDeleting = false; } @@ -39,7 +40,7 @@ export default class DaActionBar extends LitElement { } handleClear() { - this._canPaste = false; + this._isCopying = false; this._isMoving = false; this._isDeleting = false; const opts = { detail: true, bubbles: true, composed: true }; @@ -54,12 +55,12 @@ export default class DaActionBar extends LitElement { } handleCopy() { - this._canPaste = true; + this._isCopying = true; } handleMove() { + this._isCopying = true; this._isMoving = true; - this._canPaste = true; } handlePaste() { @@ -106,13 +107,19 @@ export default class DaActionBar extends LitElement { return itemDir !== this.currentPath; } + get _canWrite() { + if (!this.permissions) return false; + return this.permissions.some((permission) => permission === 'write'); + } + get _canShare() { - return this.items.some((item) => item.ext && item.ext !== 'link'); + const isFile = this.items.some((item) => item.ext && item.ext !== 'link'); + return isFile && !this._isCopying; } get currentAction() { const itemStr = this.items.length > 1 ? 'items' : 'item'; - if (this._canPaste) { + if (this._isCopying && this._canWrite) { const folderName = this.currentPath.split('/').pop(); return `Paste ${this.items.length} ${itemStr} into ${folderName}`; } @@ -135,37 +142,37 @@ export default class DaActionBar extends LitElement {
+ @click=${this.handleMove} + class="copy-button ${this._canWrite ? '' : 'hide'} ${this._isCopying ? 'hide' : ''}"> + + Cut + diff --git a/blocks/browse/da-browse/da-browse.js b/blocks/browse/da-browse/da-browse.js index dca0f052..74504f51 100644 --- a/blocks/browse/da-browse/da-browse.js +++ b/blocks/browse/da-browse/da-browse.js @@ -39,6 +39,10 @@ export default class DaBrowse extends LitElement { this.shadowRoot.adoptedStyleSheets = [STYLE]; } + handlePermissions(e) { + if (this.newCmp) this.newCmp.permissions = e.detail; + } + handleTabClick(idx) { this._tabItems = this._tabItems.map((tab, tidx) => ({ ...tab, selected: idx === tidx })); } @@ -55,6 +59,10 @@ export default class DaBrowse extends LitElement { return this._tabItems.find((tab) => tab.selected).id; } + get newCmp() { + return this.shadowRoot.querySelector('da-new'); + } + renderNew() { return html``; } @@ -68,6 +76,7 @@ export default class DaBrowse extends LitElement { `; diff --git a/blocks/browse/da-list/da-list.js b/blocks/browse/da-list/da-list.js index 2074090f..4a1f1dc6 100644 --- a/blocks/browse/da-list/da-list.js +++ b/blocks/browse/da-list/da-list.js @@ -25,6 +25,7 @@ export default class DaList extends LitElement { drag: { type: Boolean }, listItems: { attribute: false }, newItem: { attribute: false }, + _permissions: { state: true }, _listItems: { state: true }, _selectedItems: { state: true }, _dropFiles: { state: true }, @@ -51,6 +52,7 @@ export default class DaList extends LitElement { } if (props.has('fullpath') && this.fullpath) { + this.handlePermissions(null); this._listItems = await this.getList(); } @@ -69,9 +71,19 @@ export default class DaList extends LitElement { this._status = { type, text, description }; } + handlePermissions(permissions) { + this._permissions = permissions; + + // Notify parent + const opts = { detail: permissions, bubbles: true, composed: true }; + const event = new CustomEvent('onpermissions', opts); + this.dispatchEvent(event); + } + async getList() { const resp = await daFetch(`${DA_ORIGIN}/list${this.fullpath}`); - if (!resp.ok) return null; + if (!resp.ok) return []; + if (resp.permissions) this.handlePermissions(resp.permissions); return resp.json(); } @@ -127,6 +139,12 @@ export default class DaList extends LitElement { this.requestUpdate(); } + wait(milliseconds) { + return new Promise((r) => { + setTimeout(r, milliseconds); + }); + } + async handlePasteItem(item) { let continuation = true; let continuationToken; @@ -140,6 +158,13 @@ export default class DaList extends LitElement { if (resp.status === 204) { continuation = false; break; + } else if (resp.status >= 400) { + this.setStatus('Copying', 'There was an issue copying.'); + + // TODO maybe there is a better way to keep the status dialog visible for a bit? + await this.wait(2000); + + return; } const json = await resp.json(); ({ continuationToken } = json); @@ -418,6 +443,7 @@ export default class DaList extends LitElement { ${this.drag ? this.renderDropArea() : nothing}
permission === 'write'); + } + render() { return html`
- + - + `; } } diff --git a/blocks/sheet/utils/index.js b/blocks/sheet/utils/index.js index fdff7c15..80e70048 100644 --- a/blocks/sheet/utils/index.js +++ b/blocks/sheet/utils/index.js @@ -8,6 +8,9 @@ const loadScript = (await import(`${getNx()}/utils/script.js`)).default; const SHEET_TEMPLATE = { minDimensions: [20, 20], sheetName: 'data' }; +let permissions; +let canWrite; + function resetSheets(el) { document.querySelector('da-sheet-tabs')?.remove(); if (!el.jexcel) return; @@ -30,12 +33,13 @@ function finishSetup(el, data) { // Setup tabs const daSheetTabs = document.createElement('da-sheet-tabs'); + daSheetTabs.permissions = permissions; el.insertAdjacentElement('beforebegin', daSheetTabs); } function getDefaultSheet() { return [ - { ...SHEET_TEMPLATE }, + { ...SHEET_TEMPLATE, minDimensions: [20, 20] }, ]; } @@ -50,21 +54,48 @@ function getSheetData(sheetData) { return [header, ...data]; } -const getColWidths = (colWidths, headers) => colWidths?.map((width) => ({ width: `${width}` })) - || headers.map(() => ({ width: '300' })); +const getColWidths = (colWidths, headers) => { + if (colWidths) { + return colWidths?.map((width) => { + const opts = { width: `${width}` }; + opts.readOnly = !canWrite; + return opts; + }); + } + return headers.map(() => { + const opts = { width: '300' }; + opts.readOnly = !canWrite; + return opts; + }); +}; function getSheet(json, sheetName) { const data = getSheetData(json.data); + const templ = { ...SHEET_TEMPLATE }; + if (!canWrite) delete templ.minDimensions; + return { - ...SHEET_TEMPLATE, + ...templ, sheetName, data, columns: getColWidths(json[':colWidths'], data[0]), }; } +export function getPermissions() { + return permissions; +} + export async function getData(url) { const resp = await daFetch(url); + + // Set permissions even if the file is a 404 + const daTitle = document.querySelector('da-title'); + if (daTitle) daTitle.permissions = resp.permissions; + + permissions = resp.permissions; + canWrite = resp.permissions.some((permission) => permission === 'write'); + if (!resp.ok) return getDefaultSheet(); const sheets = []; diff --git a/deps/jspreadsheet-ce/dist/jspreadsheet.css b/deps/jspreadsheet-ce/dist/jspreadsheet.css index d4a1a227..3ae87a4f 100644 --- a/deps/jspreadsheet-ce/dist/jspreadsheet.css +++ b/deps/jspreadsheet-ce/dist/jspreadsheet.css @@ -1,761 +1,761 @@ -:root { - --jexcel-border-color:#000; -} - -.jexcel_container { - display:inline-block; - padding-right:2px; - box-sizing: border-box; - overscroll-behavior: contain; - outline: none; -} - -.jexcel_container.fullscreen { - position:fixed; - top:0px; - left:0px; - width:100%; - height:100%; - z-index:21; -} - -.jexcel_container.fullscreen .jexcel_content { - overflow:auto; - width:100%; - height:100%; - background-color:#ffffff; -} - -.jexcel_container.with-toolbar .jexcel > thead > tr > td { - top: 0; -} - -.jexcel_container.fullscreen.with-toolbar { - height: calc(100% - 46px); -} - -.jexcel_content { - display:inline-block; - box-sizing: border-box; - padding-right:3px; - padding-bottom:3px; - position:relative; - scrollbar-width: thin; - scrollbar-color: #666 transparent; -} - -@supports (-moz-appearance:none) { - .jexcel_content { padding-right:10px; } -} - -.jexcel_content::-webkit-scrollbar { - width: 8px; - height: 8px; -} - -.jexcel_content::-webkit-scrollbar-track { - background: #eee; -} - -.jexcel_content::-webkit-scrollbar-thumb { - background: #666; -} - -.jexcel { - border-collapse: separate; - table-layout: fixed; - white-space: nowrap; - empty-cells: show; - border: 0px; - background-color: #fff; - width: 0; - - border-top: 1px solid transparent; - border-left: 1px solid transparent; - border-right: 1px solid #ccc; - border-bottom: 1px solid #ccc; -} - -.jexcel > thead > tr > td -{ - border-top: 1px solid #ccc; - border-left: 1px solid #ccc; - border-right: 1px solid transparent; - border-bottom: 1px solid transparent; - background-color: #f3f3f3; - padding: 2px; - cursor: pointer; - box-sizing: border-box; - overflow: hidden; - position: -webkit-sticky; - position: sticky; - top: 0; - z-index:2; -} - -.jexcel_container.with-toolbar .jexcel > thead > tr > td -{ - top:42px; -} - -.jexcel > thead > tr > td.dragging -{ - background-color:#fff; - opacity:0.5; -} - -.jexcel > thead > tr > td.selected -{ - background-color:#dcdcdc; -} - -.jexcel > thead > tr > td.arrow-up -{ - background-repeat:no-repeat; - background-position:center right 5px; - background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3Cpath d='M7 14l5-5 5 5H7z' fill='gray'/%3E%3C/svg%3E"); - text-decoration:underline; -} - -.jexcel > thead > tr > td.arrow-down -{ - background-repeat:no-repeat; - background-position:center right 5px; - background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3Cpath d='M7 10l5 5 5-5H7z' fill='gray'/%3E%3C/svg%3E"); - text-decoration:underline; -} - -.jexcel > tbody > tr > td:first-child -{ - position:relative; - background-color:#f3f3f3; - text-align:center; -} - -.jexcel > tbody.resizable > tr > td:first-child::before -{ - content:'\00a0'; - width:100%; - height:3px; - position:absolute; - bottom:0px; - left:0px; - cursor:row-resize; -} - -.jexcel > tbody.draggable > tr > td:first-child::after -{ - content:'\00a0'; - width:3px; - height:100%; - position:absolute; - top:0px; - right:0px; - cursor:move; -} - -.jexcel > tbody > tr.dragging > td -{ - background-color:#eee; - opacity:0.5; -} - -.jexcel > tbody > tr > td -{ - border-top:1px solid #ccc; - border-left:1px solid #ccc; - border-right:1px solid transparent; - border-bottom:1px solid transparent; - padding:4px; - white-space: nowrap; - box-sizing: border-box; - line-height:1em; -} - -.jexcel_overflow > tbody > tr > td { - overflow: hidden; -} - -.jexcel > tbody > tr > td:last-child -{ - overflow: hidden; -} - -.jexcel > tbody > tr > td > img -{ - display:inline-block; - max-width:100px; -} - -.jexcel > tbody > tr > td.readonly -{ - color:rgba(0,0,0,0.3) -} -.jexcel > tbody > tr.selected > td:first-child -{ - background-color:#dcdcdc; -} -.jexcel > tbody > tr > td > select, -.jexcel > tbody > tr > td > input, -.jexcel > tbody > tr > td > textarea -{ - border:0px; - border-radius:0px; - outline:0px; - width:100%; - margin:0px; - padding:0px; - padding-right:2px; - background-color:transparent; - box-sizing: border-box; -} - -.jexcel > tbody > tr > td > textarea -{ - resize: none; - padding-top:6px !important; -} - -.jexcel > tbody > tr > td > input[type=checkbox] -{ - width:12px; - margin-top:2px; -} -.jexcel > tbody > tr > td > input[type=radio] -{ - width:12px; - margin-top:2px; -} - -.jexcel > tbody > tr > td > select -{ - -webkit-appearance: none; - -moz-appearance: none; - appearance: none; - background-repeat: no-repeat; - background-position-x: 100%; - background-position-y: 40%; - background-image: url(); -} - -.jexcel > tbody > tr > td.jexcel_dropdown -{ - background-repeat: no-repeat; - background-position:top 50% right 5px; - background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3Cpath d='M7 10l5 5 5-5H7z' fill='lightgray'/%3E%3C/svg%3E"); - text-overflow: ellipsis; - overflow-x:hidden; -} - -.jexcel > tbody > tr > td.jexcel_dropdown.jexcel_comments -{ - background:url("') top right no-repeat; -} - -.jexcel > tbody > tr > td > .color -{ - width:90%; - height:10px; - margin:auto; -} - -.jexcel > tbody > tr > td > a { - text-decoration: underline; -} - -.jexcel > tbody > tr > td.highlight > a { - color: blue; - cursor: pointer; -} - -.jexcel > tfoot > tr > td -{ - border-top: 1px solid #ccc; - border-left: 1px solid #ccc; - border-right: 1px solid transparent; - border-bottom: 1px solid transparent; - background-color: #f3f3f3; - padding: 2px; - cursor: pointer; - box-sizing: border-box; - overflow: hidden; -} - -.jexcel .highlight { - background-color:rgba(0,0,0,0.05); -} - -.jexcel .highlight-top { - border-top:1px solid #000; /* var(--jexcel-border-color);*/ - box-shadow: 0px -1px #ccc; -} - -.jexcel .highlight-left { - border-left:1px solid #000; /* var(--jexcel-border-color);*/ - box-shadow: -1px 0px #ccc; -} - -.jexcel .highlight-right { - border-right:1px solid #000; /* var(--jexcel-border-color);*/ -} - -.jexcel .highlight-bottom { - border-bottom:1px solid #000; /* var(--jexcel-border-color);*/ -} - -.jexcel .highlight-top.highlight-left { - box-shadow: -1px -1px #ccc; - -webkit-box-shadow: -1px -1px #ccc; - -moz-box-shadow: -1px -1px #ccc; -} - -.jexcel .highlight-selected -{ - background-color:rgba(0,0,0,0.0); -} -.jexcel .selection -{ - background-color:rgba(0,0,0,0.05); -} -.jexcel .selection-left -{ - border-left:1px dotted #000; -} -.jexcel .selection-right -{ - border-right:1px dotted #000; -} -.jexcel .selection-top -{ - border-top:1px dotted #000; -} -.jexcel .selection-bottom -{ - border-bottom:1px dotted #000; -} -.jexcel_corner -{ - position:absolute; - background-color: rgb(0, 0, 0); - height: 1px; - width: 1px; - border: 1px solid rgb(255, 255, 255); - top:-2000px; - left:-2000px; - cursor:crosshair; - box-sizing: initial; - z-index:20; - padding: 2px; -} - -.jexcel .editor -{ - outline:0px solid transparent; - overflow:visible; - white-space: nowrap; - text-align:left; - padding:0px; - box-sizing: border-box; - overflow:visible !important; -} - -.jexcel .editor > input -{ - padding-left:4px; -} - -.jexcel .editor .jupload -{ - position:fixed; - top:100%; - z-index:40; - user-select:none; - -webkit-font-smoothing: antialiased; - font-size: .875rem; - letter-spacing: .2px; - -webkit-border-radius: 4px; - border-radius: 4px; - -webkit-box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2); - box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2); - padding:10px; - background-color:#fff; - width:300px; - min-height:225px; - margin-top:2px; -} - -.jexcel .editor .jupload img -{ - width:100%; - height:auto; -} - -.jexcel .editor .jexcel_richtext -{ - position:fixed; - top:100%; - z-index:40; - user-select:none; - -webkit-font-smoothing: antialiased; - font-size: .875rem; - letter-spacing: .2px; - -webkit-box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2); - box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2); - padding:10px; - background-color:#fff; - min-width:280px; - max-width:310px; - margin-top:2px; - text-align:left; -} - -.jexcel .editor .jclose:after -{ - position:absolute; - top:0; - right:0; - margin:10px; - content:'close'; - font-family:'Material icons'; - font-size:24px; - width:24px; - height:24px; - line-height:24px; - cursor:pointer; - text-shadow: 0px 0px 5px #fff; -} - -.jexcel, .jexcel td, .jexcel_corner -{ - -webkit-touch-callout: none; - -webkit-user-select: none; - -khtml-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - -webkit-user-drag: none; - -khtml-user-drag: none; - -moz-user-drag: none; - -o-user-drag: none; - user-drag: none; -} - -.jexcel_textarea -{ - position:absolute; - top:-999px; - left:-999px; - width:1px; - height:1px; -} -.jexcel .dragline -{ - position:absolute; -} -.jexcel .dragline div -{ - position:relative; - top:-6px; - height:5px; - width:22px; -} -.jexcel .dragline div:hover -{ - cursor:move; -} - -.jexcel .onDrag -{ - background-color:rgba(0,0,0,0.6); -} - -.jexcel .error -{ - border:1px solid red; -} - -.jexcel thead td.resizing -{ - border-right-style:dotted !important; - border-right-color:red !important; -} - -.jexcel tbody tr.resizing > td -{ - border-bottom-style:dotted !important; - border-bottom-color:red !important; -} - -.jexcel tbody td.resizing -{ - border-right-style:dotted !important; - border-right-color:red !important; -} - -.jexcel .jdropdown-header -{ - border:0px !important; - outline:none !important; - width:100% !important; - height:100% !important; - padding:0px !important; - padding-left:8px !important; -} - -.jexcel .jdropdown-container -{ - margin-top:1px; -} - -.jexcel .jdropdown-container-header { - padding: 0px; - margin: 0px; - height: inherit; -} - -.jexcel .jdropdown-picker -{ - border:0px !important; - padding:0px !important; - width:inherit; - height:inherit; -} - -.jexcel .jexcel_comments -{ - background:url(''); - background-repeat: no-repeat; - background-position: top right; -} - -.jexcel .sp-replacer -{ - margin: 2px; - border:0px; -} - -.jexcel > thead > tr.jexcel_filter > td > input -{ - border:0px; - width:100%; - outline:none; -} - -.jexcel_about { - float: right; - font-size: 0.7em; - padding: 2px; - text-transform: uppercase; - letter-spacing: 1px; - display: none; -} -.jexcel_about a { - color: #ccc; - text-decoration: none; -} - -.jexcel_about img { - display: none; -} - -.jexcel_filter -{ - display:flex; - justify-content:space-between; - margin-bottom:4px; -} - -.jexcel_filter > div -{ - padding:8px; - align-items:center; -} - -.jexcel_pagination -{ - display:flex; - justify-content:space-between; - align-items:center; -} - -.jexcel_pagination > div -{ - display:flex; - padding:10px; -} - -.jexcel_pagination > div:last-child -{ - padding-right:10px; - padding-top:10px; -} - -.jexcel_pagination > div > div -{ - text-align:center; - width:36px; - height:36px; - line-height:34px; - border:1px solid #ccc; - box-sizing: border-box; - margin-left:2px; - cursor:pointer; -} - -.jexcel_page -{ - font-size:0.8em; -} - -.jexcel_page_selected -{ - font-weight:bold; - background-color:#f3f3f3; -} - -.jexcel_toolbar -{ - display:flex; - background-color:#f3f3f3; - border:1px solid #ccc; - padding:4px; - margin:0px 2px 4px 1px; - position:sticky; - top:0px; - z-index:21; -} - -.jexcel_toolbar:empty -{ - display:none; -} - -.jexcel_toolbar i.jexcel_toolbar_item -{ - width:24px; - height:24px; - padding:4px; - cursor:pointer; - display:inline-block; -} - -.jexcel_toolbar i.jexcel_toolbar_item:hover -{ - background-color:#ddd; -} - -.jexcel_toolbar select.jexcel_toolbar_item -{ - margin-left:2px; - margin-right:2px; - display:inline-block; - border:0px; - background-color:transparent; - padding-right:10px; -} - -.jexcel .dragging-left -{ - background-repeat: no-repeat; - background-position:top 50% left 0px; - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M14 7l-5 5 5 5V7z'/%3E%3Cpath fill='none' d='M24 0v24H0V0h24z'/%3E%3C/svg%3E"); -} - -.jexcel .dragging-right -{ - background-repeat: no-repeat; - background-position:top 50% right 0px; - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M10 17l5-5-5-5v10z'/%3E%3Cpath fill='none' d='M0 24V0h24v24H0z'/%3E%3C/svg%3E"); -} - -.jexcel_tabs .jexcel_tab -{ - display:none; -} - -.jexcel_tabs .jexcel_tab_link -{ - display:inline-block; - padding:10px; - padding-left:20px; - padding-right:20px; - margin-right:5px; - margin-bottom:5px; - background-color:#f3f3f3; - cursor:pointer; -} - -.jexcel_tabs .jexcel_tab_link.selected -{ - background-color:#ddd; -} - -.jexcel_hidden_index > tbody > tr > td:first-child, -.jexcel_hidden_index > thead > tr > td:first-child, -.jexcel_hidden_index > tfoot > tr > td:first-child, -.jexcel_hidden_index > colgroup > col:first-child -{ - display:none; -} - - - -.jexcel .jrating { - display: inline-flex; -} -.jexcel .jrating > div { - zoom: 0.55; -} - -.jexcel .copying-top { - border-top:1px dashed #000; -} - -.jexcel .copying-left { - border-left:1px dashed #000; -} - -.jexcel .copying-right { - border-right:1px dashed #000; -} - -.jexcel .copying-bottom { - border-bottom:1px dashed #000; -} - -.jexcel .jexcel_column_filter { - background-repeat: no-repeat; - background-position: top 50% right 5px; - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' fill='gray' width='18px' height='18px'%3E%3Cpath d='M10 18h4v-2h-4v2zM3 6v2h18V6H3zm3 7h12v-2H6v2z'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E"); - text-overflow: ellipsis; - overflow: hidden; - padding: 0px; - padding-left: 6px; - padding-right: 20px; -} - -.jexcel thead .jexcel_freezed, .jexcel tfoot .jexcel_freezed { - left: 0px; - z-index: 3 !important; - box-shadow: 2px 0px 2px 0.2px #ccc !important; - -webkit-box-shadow: 2px 0px 2px 0.2px #ccc !important; - -moz-box-shadow: 2px 0px 2px 0.2px #ccc !important; -} - -.jexcel tbody .jexcel_freezed { - position: relative; - background-color: #fff; - box-shadow: 1px 1px 1px 1px #ccc !important; - -webkit-box-shadow: 2px 4px 4px 0.1px #ccc !important; - -moz-box-shadow: 2px 4px 4px 0.1px #ccc !important; -} - -.red { - color: red; -} - -.jexcel > tbody > tr > td.readonly > input[type=checkbox], -.jexcel > tbody > tr > td.readonly > input[type=radio] { - pointer-events: none; - opacity: 0.5; -} \ No newline at end of file +:root { + --jexcel-border-color:#000; +} + +.jexcel_container { + display:inline-block; + padding-right:2px; + box-sizing: border-box; + overscroll-behavior: contain; + outline: none; +} + +.jexcel_container.fullscreen { + position:fixed; + top:0px; + left:0px; + width:100%; + height:100%; + z-index:21; +} + +.jexcel_container.fullscreen .jexcel_content { + overflow:auto; + width:100%; + height:100%; + background-color:#ffffff; +} + +.jexcel_container.with-toolbar .jexcel > thead > tr > td { + top: 0; +} + +.jexcel_container.fullscreen.with-toolbar { + height: calc(100% - 46px); +} + +.jexcel_content { + display:inline-block; + box-sizing: border-box; + padding-right:3px; + padding-bottom:3px; + position:relative; + scrollbar-width: thin; + scrollbar-color: #666 transparent; +} + +@supports (-moz-appearance:none) { + .jexcel_content { padding-right:10px; } +} + +.jexcel_content::-webkit-scrollbar { + width: 8px; + height: 8px; +} + +.jexcel_content::-webkit-scrollbar-track { + background: #eee; +} + +.jexcel_content::-webkit-scrollbar-thumb { + background: #666; +} + +.jexcel { + border-collapse: separate; + table-layout: fixed; + white-space: nowrap; + empty-cells: show; + border: 0px; + background-color: #fff; + width: 0; + + border-top: 1px solid transparent; + border-left: 1px solid transparent; + border-right: 1px solid #ccc; + border-bottom: 1px solid #ccc; +} + +.jexcel > thead > tr > td +{ + border-top: 1px solid #ccc; + border-left: 1px solid #ccc; + border-right: 1px solid transparent; + border-bottom: 1px solid transparent; + background-color: #f3f3f3; + padding: 2px; + cursor: pointer; + box-sizing: border-box; + overflow: hidden; + position: -webkit-sticky; + position: sticky; + top: 0; + z-index:2; +} + +.jexcel_container.with-toolbar .jexcel > thead > tr > td +{ + top:42px; +} + +.jexcel > thead > tr > td.dragging +{ + background-color:#fff; + opacity:0.5; +} + +.jexcel > thead > tr > td.selected +{ + background-color:#dcdcdc; +} + +.jexcel > thead > tr > td.arrow-up +{ + background-repeat:no-repeat; + background-position:center right 5px; + background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3Cpath d='M7 14l5-5 5 5H7z' fill='gray'/%3E%3C/svg%3E"); + text-decoration:underline; +} + +.jexcel > thead > tr > td.arrow-down +{ + background-repeat:no-repeat; + background-position:center right 5px; + background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3Cpath d='M7 10l5 5 5-5H7z' fill='gray'/%3E%3C/svg%3E"); + text-decoration:underline; +} + +.jexcel > tbody > tr > td:first-child +{ + position:relative; + background-color:#f3f3f3; + text-align:center; +} + +.jexcel > tbody.resizable > tr > td:first-child::before +{ + content:'\00a0'; + width:100%; + height:3px; + position:absolute; + bottom:0px; + left:0px; + cursor:row-resize; +} + +.jexcel > tbody.draggable > tr > td:first-child::after +{ + content:'\00a0'; + width:3px; + height:100%; + position:absolute; + top:0px; + right:0px; + cursor:move; +} + +.jexcel > tbody > tr.dragging > td +{ + background-color:#eee; + opacity:0.5; +} + +.jexcel > tbody > tr > td +{ + border-top:1px solid #ccc; + border-left:1px solid #ccc; + border-right:1px solid transparent; + border-bottom:1px solid transparent; + padding:4px; + white-space: nowrap; + box-sizing: border-box; + line-height:1em; +} + +.jexcel_overflow > tbody > tr > td { + overflow: hidden; +} + +.jexcel > tbody > tr > td:last-child +{ + overflow: hidden; +} + +.jexcel > tbody > tr > td > img +{ + display:inline-block; + max-width:100px; +} + +.jexcel > tbody > tr > td.readonly +{ + background-color: #f9f9f9; +} +.jexcel > tbody > tr.selected > td:first-child +{ + background-color:#dcdcdc; +} +.jexcel > tbody > tr > td > select, +.jexcel > tbody > tr > td > input, +.jexcel > tbody > tr > td > textarea +{ + border:0px; + border-radius:0px; + outline:0px; + width:100%; + margin:0px; + padding:0px; + padding-right:2px; + background-color:transparent; + box-sizing: border-box; +} + +.jexcel > tbody > tr > td > textarea +{ + resize: none; + padding-top:6px !important; +} + +.jexcel > tbody > tr > td > input[type=checkbox] +{ + width:12px; + margin-top:2px; +} +.jexcel > tbody > tr > td > input[type=radio] +{ + width:12px; + margin-top:2px; +} + +.jexcel > tbody > tr > td > select +{ + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + background-repeat: no-repeat; + background-position-x: 100%; + background-position-y: 40%; + background-image: url(); +} + +.jexcel > tbody > tr > td.jexcel_dropdown +{ + background-repeat: no-repeat; + background-position:top 50% right 5px; + background-image: url("data:image/svg+xml,%0A%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath fill='none' d='M0 0h24v24H0V0z'/%3E%3Cpath d='M7 10l5 5 5-5H7z' fill='lightgray'/%3E%3C/svg%3E"); + text-overflow: ellipsis; + overflow-x:hidden; +} + +.jexcel > tbody > tr > td.jexcel_dropdown.jexcel_comments +{ + background:url("') top right no-repeat; +} + +.jexcel > tbody > tr > td > .color +{ + width:90%; + height:10px; + margin:auto; +} + +.jexcel > tbody > tr > td > a { + text-decoration: underline; +} + +.jexcel > tbody > tr > td.highlight > a { + color: blue; + cursor: pointer; +} + +.jexcel > tfoot > tr > td +{ + border-top: 1px solid #ccc; + border-left: 1px solid #ccc; + border-right: 1px solid transparent; + border-bottom: 1px solid transparent; + background-color: #f3f3f3; + padding: 2px; + cursor: pointer; + box-sizing: border-box; + overflow: hidden; +} + +.jexcel .highlight { + background-color:rgba(0,0,0,0.05); +} + +.jexcel .highlight-top { + border-top:1px solid #000; /* var(--jexcel-border-color);*/ + box-shadow: 0px -1px #ccc; +} + +.jexcel .highlight-left { + border-left:1px solid #000; /* var(--jexcel-border-color);*/ + box-shadow: -1px 0px #ccc; +} + +.jexcel .highlight-right { + border-right:1px solid #000; /* var(--jexcel-border-color);*/ +} + +.jexcel .highlight-bottom { + border-bottom:1px solid #000; /* var(--jexcel-border-color);*/ +} + +.jexcel .highlight-top.highlight-left { + box-shadow: -1px -1px #ccc; + -webkit-box-shadow: -1px -1px #ccc; + -moz-box-shadow: -1px -1px #ccc; +} + +.jexcel .highlight-selected +{ + background-color:rgba(0,0,0,0.0); +} +.jexcel .selection +{ + background-color:rgba(0,0,0,0.05); +} +.jexcel .selection-left +{ + border-left:1px dotted #000; +} +.jexcel .selection-right +{ + border-right:1px dotted #000; +} +.jexcel .selection-top +{ + border-top:1px dotted #000; +} +.jexcel .selection-bottom +{ + border-bottom:1px dotted #000; +} +.jexcel_corner +{ + position:absolute; + background-color: rgb(0, 0, 0); + height: 1px; + width: 1px; + border: 1px solid rgb(255, 255, 255); + top:-2000px; + left:-2000px; + cursor:crosshair; + box-sizing: initial; + z-index:20; + padding: 2px; +} + +.jexcel .editor +{ + outline:0px solid transparent; + overflow:visible; + white-space: nowrap; + text-align:left; + padding:0px; + box-sizing: border-box; + overflow:visible !important; +} + +.jexcel .editor > input +{ + padding-left:4px; +} + +.jexcel .editor .jupload +{ + position:fixed; + top:100%; + z-index:40; + user-select:none; + -webkit-font-smoothing: antialiased; + font-size: .875rem; + letter-spacing: .2px; + -webkit-border-radius: 4px; + border-radius: 4px; + -webkit-box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2); + box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2); + padding:10px; + background-color:#fff; + width:300px; + min-height:225px; + margin-top:2px; +} + +.jexcel .editor .jupload img +{ + width:100%; + height:auto; +} + +.jexcel .editor .jexcel_richtext +{ + position:fixed; + top:100%; + z-index:40; + user-select:none; + -webkit-font-smoothing: antialiased; + font-size: .875rem; + letter-spacing: .2px; + -webkit-box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2); + box-shadow: 0 8px 10px 1px rgba(0,0,0,0.14), 0 3px 14px 2px rgba(0,0,0,0.12), 0 5px 5px -3px rgba(0,0,0,0.2); + padding:10px; + background-color:#fff; + min-width:280px; + max-width:310px; + margin-top:2px; + text-align:left; +} + +.jexcel .editor .jclose:after +{ + position:absolute; + top:0; + right:0; + margin:10px; + content:'close'; + font-family:'Material icons'; + font-size:24px; + width:24px; + height:24px; + line-height:24px; + cursor:pointer; + text-shadow: 0px 0px 5px #fff; +} + +.jexcel, .jexcel td, .jexcel_corner +{ + -webkit-touch-callout: none; + -webkit-user-select: none; + -khtml-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + -webkit-user-drag: none; + -khtml-user-drag: none; + -moz-user-drag: none; + -o-user-drag: none; + user-drag: none; +} + +.jexcel_textarea +{ + position:absolute; + top:-999px; + left:-999px; + width:1px; + height:1px; +} +.jexcel .dragline +{ + position:absolute; +} +.jexcel .dragline div +{ + position:relative; + top:-6px; + height:5px; + width:22px; +} +.jexcel .dragline div:hover +{ + cursor:move; +} + +.jexcel .onDrag +{ + background-color:rgba(0,0,0,0.6); +} + +.jexcel .error +{ + border:1px solid red; +} + +.jexcel thead td.resizing +{ + border-right-style:dotted !important; + border-right-color:red !important; +} + +.jexcel tbody tr.resizing > td +{ + border-bottom-style:dotted !important; + border-bottom-color:red !important; +} + +.jexcel tbody td.resizing +{ + border-right-style:dotted !important; + border-right-color:red !important; +} + +.jexcel .jdropdown-header +{ + border:0px !important; + outline:none !important; + width:100% !important; + height:100% !important; + padding:0px !important; + padding-left:8px !important; +} + +.jexcel .jdropdown-container +{ + margin-top:1px; +} + +.jexcel .jdropdown-container-header { + padding: 0px; + margin: 0px; + height: inherit; +} + +.jexcel .jdropdown-picker +{ + border:0px !important; + padding:0px !important; + width:inherit; + height:inherit; +} + +.jexcel .jexcel_comments +{ + background:url(''); + background-repeat: no-repeat; + background-position: top right; +} + +.jexcel .sp-replacer +{ + margin: 2px; + border:0px; +} + +.jexcel > thead > tr.jexcel_filter > td > input +{ + border:0px; + width:100%; + outline:none; +} + +.jexcel_about { + float: right; + font-size: 0.7em; + padding: 2px; + text-transform: uppercase; + letter-spacing: 1px; + display: none; +} +.jexcel_about a { + color: #ccc; + text-decoration: none; +} + +.jexcel_about img { + display: none; +} + +.jexcel_filter +{ + display:flex; + justify-content:space-between; + margin-bottom:4px; +} + +.jexcel_filter > div +{ + padding:8px; + align-items:center; +} + +.jexcel_pagination +{ + display:flex; + justify-content:space-between; + align-items:center; +} + +.jexcel_pagination > div +{ + display:flex; + padding:10px; +} + +.jexcel_pagination > div:last-child +{ + padding-right:10px; + padding-top:10px; +} + +.jexcel_pagination > div > div +{ + text-align:center; + width:36px; + height:36px; + line-height:34px; + border:1px solid #ccc; + box-sizing: border-box; + margin-left:2px; + cursor:pointer; +} + +.jexcel_page +{ + font-size:0.8em; +} + +.jexcel_page_selected +{ + font-weight:bold; + background-color:#f3f3f3; +} + +.jexcel_toolbar +{ + display:flex; + background-color:#f3f3f3; + border:1px solid #ccc; + padding:4px; + margin:0px 2px 4px 1px; + position:sticky; + top:0px; + z-index:21; +} + +.jexcel_toolbar:empty +{ + display:none; +} + +.jexcel_toolbar i.jexcel_toolbar_item +{ + width:24px; + height:24px; + padding:4px; + cursor:pointer; + display:inline-block; +} + +.jexcel_toolbar i.jexcel_toolbar_item:hover +{ + background-color:#ddd; +} + +.jexcel_toolbar select.jexcel_toolbar_item +{ + margin-left:2px; + margin-right:2px; + display:inline-block; + border:0px; + background-color:transparent; + padding-right:10px; +} + +.jexcel .dragging-left +{ + background-repeat: no-repeat; + background-position:top 50% left 0px; + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M14 7l-5 5 5 5V7z'/%3E%3Cpath fill='none' d='M24 0v24H0V0h24z'/%3E%3C/svg%3E"); +} + +.jexcel .dragging-right +{ + background-repeat: no-repeat; + background-position:top 50% right 0px; + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cpath d='M10 17l5-5-5-5v10z'/%3E%3Cpath fill='none' d='M0 24V0h24v24H0z'/%3E%3C/svg%3E"); +} + +.jexcel_tabs .jexcel_tab +{ + display:none; +} + +.jexcel_tabs .jexcel_tab_link +{ + display:inline-block; + padding:10px; + padding-left:20px; + padding-right:20px; + margin-right:5px; + margin-bottom:5px; + background-color:#f3f3f3; + cursor:pointer; +} + +.jexcel_tabs .jexcel_tab_link.selected +{ + background-color:#ddd; +} + +.jexcel_hidden_index > tbody > tr > td:first-child, +.jexcel_hidden_index > thead > tr > td:first-child, +.jexcel_hidden_index > tfoot > tr > td:first-child, +.jexcel_hidden_index > colgroup > col:first-child +{ + display:none; +} + + + +.jexcel .jrating { + display: inline-flex; +} +.jexcel .jrating > div { + zoom: 0.55; +} + +.jexcel .copying-top { + border-top:1px dashed #000; +} + +.jexcel .copying-left { + border-left:1px dashed #000; +} + +.jexcel .copying-right { + border-right:1px dashed #000; +} + +.jexcel .copying-bottom { + border-bottom:1px dashed #000; +} + +.jexcel .jexcel_column_filter { + background-repeat: no-repeat; + background-position: top 50% right 5px; + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' fill='gray' width='18px' height='18px'%3E%3Cpath d='M10 18h4v-2h-4v2zM3 6v2h18V6H3zm3 7h12v-2H6v2z'/%3E%3Cpath d='M0 0h24v24H0z' fill='none'/%3E%3C/svg%3E"); + text-overflow: ellipsis; + overflow: hidden; + padding: 0px; + padding-left: 6px; + padding-right: 20px; +} + +.jexcel thead .jexcel_freezed, .jexcel tfoot .jexcel_freezed { + left: 0px; + z-index: 3 !important; + box-shadow: 2px 0px 2px 0.2px #ccc !important; + -webkit-box-shadow: 2px 0px 2px 0.2px #ccc !important; + -moz-box-shadow: 2px 0px 2px 0.2px #ccc !important; +} + +.jexcel tbody .jexcel_freezed { + position: relative; + background-color: #fff; + box-shadow: 1px 1px 1px 1px #ccc !important; + -webkit-box-shadow: 2px 4px 4px 0.1px #ccc !important; + -moz-box-shadow: 2px 4px 4px 0.1px #ccc !important; +} + +.red { + color: red; +} + +.jexcel > tbody > tr > td.readonly > input[type=checkbox], +.jexcel > tbody > tr > td.readonly > input[type=radio] { + pointer-events: none; + opacity: 0.5; +} diff --git a/test/e2e/tests/delete.spec.js b/test/e2e/tests/delete.spec.js index e1ff5fd6..f8546d3a 100644 --- a/test/e2e/tests/delete.spec.js +++ b/test/e2e/tests/delete.spec.js @@ -11,7 +11,7 @@ */ import { test, expect } from '@playwright/test'; import ENV from '../utils/env.js'; -import { getTestPageURL, getTestResourceAge } from '../utils/page.js'; +import { getQuery, getTestPageURL, getTestResourceAge } from '../utils/page.js'; // Files are deleted after 2 hours by default const MIN_HOURS = process.env.PW_DELETE_HOURS ? Number(process.env.PW_DELETE_HOURS) : 2; @@ -104,7 +104,7 @@ test('Empty out open editors on deleted documents', async ({ browser, page }, wo await page.close(); const list = await browser.newPage(); - await list.goto(`${ENV}/#/da-sites/da-status/tests`); + await list.goto(`${ENV}/${getQuery()}#/da-sites/da-status/tests`); await list.waitForTimeout(3000); await list.reload(); diff --git a/test/e2e/tests/edit.spec.js b/test/e2e/tests/edit.spec.js index d8369c4e..ed336a94 100644 --- a/test/e2e/tests/edit.spec.js +++ b/test/e2e/tests/edit.spec.js @@ -11,7 +11,7 @@ */ import { test, expect } from '@playwright/test'; import ENV from '../utils/env.js'; -import { getTestPageURL } from '../utils/page.js'; +import { getQuery, getTestPageURL } from '../utils/page.js'; test('Update Document', async ({ browser, page }, workerInfo) => { test.setTimeout(30000); @@ -40,7 +40,7 @@ test('Create Delete Document', async ({ browser, page }, workerInfo) => { const url = getTestPageURL('edit2', workerInfo); const pageName = url.split('/').pop(); - await page.goto(`${ENV}/#/da-sites/da-status/tests`); + await page.goto(`${ENV}/${getQuery()}#/da-sites/da-status/tests`); await page.locator('button.da-actions-new-button').click(); await page.locator('button:text("Document")').click(); await page.locator('input.da-actions-input').fill(pageName); @@ -51,7 +51,7 @@ test('Create Delete Document', async ({ browser, page }, workerInfo) => { await page.waitForTimeout(1000); const newPage = await browser.newPage(); - await newPage.goto(`${ENV}/#/da-sites/da-status/tests`); + await newPage.goto(`${ENV}/${getQuery()}#/da-sites/da-status/tests`); await newPage.waitForTimeout(3000); await newPage.reload(); diff --git a/test/e2e/utils/page.js b/test/e2e/utils/page.js index 5bfccca5..1bbceb9c 100644 --- a/test/e2e/utils/page.js +++ b/test/e2e/utils/page.js @@ -11,7 +11,7 @@ */ import ENV from './env.js'; -function getQuery() { +export function getQuery() { const { GITHUB_HEAD_REF: branch } = process.env; if (branch === 'local') { return '?da-admin=local&da-collab=local';