diff --git a/abstract/UploaderBlock.js b/abstract/UploaderBlock.js index 3fd7a7079..3e5b63c34 100644 --- a/abstract/UploaderBlock.js +++ b/abstract/UploaderBlock.js @@ -17,6 +17,7 @@ import { uploaderBlockCtx } from './CTX.js'; import { EVENT_TYPES, EventData, EventManager } from './EventManager.js'; import { TypedCollection } from './TypedCollection.js'; import { uploadEntrySchema } from './uploadEntrySchema.js'; +import { serializeCsv } from '../blocks/utils/comma-separated.js'; export class UploaderBlock extends ActivityBlock { couldBeUploadCollectionOwner = false; @@ -204,7 +205,7 @@ export class UploaderBlock extends ActivityBlock { /** @param {{ captureCamera?: boolean }} options */ openSystemDialog(options = {}) { - let accept = mergeFileTypes([this.cfg.accept ?? '', ...(this.cfg.imgOnly ? IMAGE_ACCEPT_LIST : [])]).join(','); + let accept = serializeCsv(mergeFileTypes([this.cfg.accept ?? '', ...(this.cfg.imgOnly ? IMAGE_ACCEPT_LIST : [])])); if (this.cfg.accept && !!this.cfg.imgOnly) { console.warn( @@ -218,7 +219,7 @@ export class UploaderBlock extends ActivityBlock { this.fileInput.multiple = this.cfg.multiple; if (options.captureCamera) { this.fileInput.capture = ''; - this.fileInput.accept = IMAGE_ACCEPT_LIST.join(','); + this.fileInput.accept = serializeCsv(IMAGE_ACCEPT_LIST); } else { this.fileInput.accept = accept; } diff --git a/blocks/CloudImageEditor/src/CloudImageEditorBlock.js b/blocks/CloudImageEditor/src/CloudImageEditorBlock.js index 60247e2df..50e398b88 100644 --- a/blocks/CloudImageEditor/src/CloudImageEditorBlock.js +++ b/blocks/CloudImageEditor/src/CloudImageEditorBlock.js @@ -12,6 +12,7 @@ import { CloudImageEditorBase } from './CloudImageEditorBase.js'; import { classNames } from './lib/classNames.js'; import { parseCropPreset } from './lib/parseCropPreset.js'; import { operationsToTransformations, transformationsToOperations } from './lib/transformationUtils.js'; +import { parseTabs } from './lib/parseTabs.js'; import { initState } from './state.js'; import { TEMPLATE } from './template.js'; import { TabId } from './toolbar-constants.js'; @@ -159,6 +160,13 @@ export class CloudImageEditorBlock extends CloudImageEditorBase { this.$['*cropPresetList'] = parseCropPreset(val); }); + this.sub( + 'tabs', + /** @param {string} val */ (val) => { + this.$['*tabList'] = parseTabs(val); + } + ); + this.sub('*tabId', (tabId) => { this.ref['img-el'].className = classNames('image', { image_hidden_to_cropper: tabId === TabId.CROP, @@ -211,4 +219,5 @@ CloudImageEditorBlock.bindAttributes({ uuid: 'uuid', 'cdn-url': 'cdnUrl', 'crop-preset': 'cropPreset', + tabs: 'tabs', }); diff --git a/blocks/CloudImageEditor/src/EditorToolbar.js b/blocks/CloudImageEditor/src/EditorToolbar.js index b060eb205..6080d0acc 100644 --- a/blocks/CloudImageEditor/src/EditorToolbar.js +++ b/blocks/CloudImageEditor/src/EditorToolbar.js @@ -1,3 +1,4 @@ +// @ts-check import { debounce } from '../../utils/debounce.js'; import { CloudImageEditorBase } from './CloudImageEditorBase.js'; import { EditorCropButtonControl } from './EditorCropButtonControl.js'; @@ -9,24 +10,26 @@ import { ALL_COLOR_OPERATIONS, ALL_CROP_OPERATIONS, ALL_FILTERS, + ALL_TABS, COLOR_OPERATIONS_CONFIG, TabId, - TABS, } from './toolbar-constants.js'; import { viewerImageSrc } from './util.js'; /** @param {String} id */ function renderTabToggle(id) { return /* HTML */ ` - - + + + + `; } @@ -67,9 +70,13 @@ export class EditorToolbar extends CloudImageEditorBase { 'presence.mainToolbar': true, 'presence.subToolbar': false, + 'presence.tabToggles': true, 'presence.tabContent.crop': false, - 'presence.tabContent.sliders': false, + 'presence.tabContent.tuning': false, 'presence.tabContent.filters': false, + 'presence.tabToggle.crop': true, + 'presence.tabToggle.tuning': true, + 'presence.tabToggle.filters': true, 'presence.subTopToolbarStyles': { hidden: 'sub-toolbar--top-hidden', visible: 'sub-toolbar--visible', @@ -82,24 +89,35 @@ export class EditorToolbar extends CloudImageEditorBase { hidden: 'tab-content--hidden', visible: 'tab-content--visible', }, - 'on.cancel': (e) => { + 'presence.tabToggleStyles': { + hidden: 'tab-toggle--hidden', + visible: 'tab-toggle--visible', + }, + 'presence.tabTogglesStyles': { + hidden: 'tab-toggles--hidden', + visible: 'tab-toggles--visible', + }, + 'on.cancel': () => { this._cancelPreload && this._cancelPreload(); this.$['*on.cancel'](); }, - 'on.apply': (e) => { + 'on.apply': () => { this.$['*on.apply'](this.$['*editorTransformations']); }, - 'on.applySlider': (e) => { + 'on.applySlider': () => { this.ref['slider-el'].apply(); this._onSliderClose(); }, - 'on.cancelSlider': (e) => { + 'on.cancelSlider': () => { this.ref['slider-el'].cancel(); this._onSliderClose(); }, + /** @param {MouseEvent} e */ 'on.clickTab': (e) => { - let id = e.currentTarget.getAttribute('data-id'); - this._activateTab(id, { fromViewer: false }); + const id = /** @type {HTMLElement} */ (e.currentTarget).getAttribute('data-id'); + if (id) { + this._activateTab(id, { fromViewer: false }); + } }, }; @@ -111,7 +129,7 @@ export class EditorToolbar extends CloudImageEditorBase { /** @private */ _onSliderClose() { this.$['*showSlider'] = false; - if (this.$['*tabId'] === TabId.SLIDERS) { + if (this.$['*tabId'] === TabId.TUNING) { this.ref['tooltip-el'].classList.toggle('info-tooltip_visible', false); } } @@ -121,8 +139,9 @@ export class EditorToolbar extends CloudImageEditorBase { * @param {String} operation */ _createOperationControl(operation) { - let el = EditorOperationControl.is && new EditorOperationControl(); - el['operation'] = operation; + let el = new EditorOperationControl(); + // @ts-expect-error TODO: fix + el.operation = operation; return el; } @@ -131,8 +150,9 @@ export class EditorToolbar extends CloudImageEditorBase { * @param {String} filter */ _createFilterControl(filter) { - let el = EditorFilterControl.is && new EditorFilterControl(); - el['filter'] = filter; + let el = new EditorFilterControl(); + // @ts-expect-error TODO: fix + el.filter = filter; return el; } @@ -141,8 +161,9 @@ export class EditorToolbar extends CloudImageEditorBase { * @param {String} operation */ _createToggleControl(operation) { - let el = EditorCropButtonControl.is && new EditorCropButtonControl(); - el['operation'] = operation; + let el = new EditorCropButtonControl(); + // @ts-expect-error TODO: fix + el.operation = operation; return el; } @@ -155,26 +176,30 @@ export class EditorToolbar extends CloudImageEditorBase { let fr = document.createDocumentFragment(); if (tabId === TabId.CROP) { - this.$.cropOperations.forEach((operation) => { - let el = this._createToggleControl(operation); - // @ts-ignore - fr.appendChild(el); - }); + this.$.cropOperations.forEach( + /** @param {string} operation */ (operation) => { + let el = this._createToggleControl(operation); + // @ts-ignore + fr.appendChild(el); + } + ); } else if (tabId === TabId.FILTERS) { [FAKE_ORIGINAL_FILTER, ...this.$.filters].forEach((filterId) => { let el = this._createFilterControl(filterId); // @ts-ignore fr.appendChild(el); }); - } else if (tabId === TabId.SLIDERS) { - this.$.colorOperations.forEach((operation) => { - let el = this._createOperationControl(operation); - // @ts-ignore - fr.appendChild(el); - }); + } else if (tabId === TabId.TUNING) { + this.$.colorOperations.forEach( + /** @param {string} operation */ (operation) => { + let el = this._createOperationControl(operation); + // @ts-ignore + fr.appendChild(el); + } + ); } - fr.childNodes.forEach((/** @type {HTMLElement} */ el, idx) => { + [...fr.children].forEach((el, idx) => { if (idx === fr.childNodes.length - 1) { el.classList.add('controls-list_last-item'); } @@ -200,7 +225,7 @@ export class EditorToolbar extends CloudImageEditorBase { this.$['*cropperEl'].deactivate(); } - for (let tabId of TABS) { + for (let tabId of ALL_TABS) { let isCurrentTab = tabId === id; let tabToggleEl = this.ref[`tab-toggle-${tabId}`]; @@ -248,13 +273,18 @@ export class EditorToolbar extends CloudImageEditorBase { } } - /** @private */ + /** + * @private + * @param {boolean} show + */ _showLoader(show) { this.$.showLoader = show; } _updateInfoTooltip = debounce(() => { - let transformations = this.$['*editorTransformations']; + const transformations = this.$['*editorTransformations']; + /** @type {keyof COLOR_OPERATIONS_CONFIG} */ + const currentOperation = this.$['*currentOperation']; let text = ''; let visible = false; @@ -266,11 +296,10 @@ export class EditorToolbar extends CloudImageEditorBase { } else { text = this.l10n(FAKE_ORIGINAL_FILTER); } - } else if (this.$['*tabId'] === TabId.SLIDERS && this.$['*currentOperation']) { + } else if (this.$['*tabId'] === TabId.TUNING && currentOperation) { visible = true; - let value = - transformations?.[this.$['*currentOperation']] || COLOR_OPERATIONS_CONFIG[this.$['*currentOperation']].zero; - text = this.$['*currentOperation'] + ' ' + value; + let value = transformations?.[currentOperation] || COLOR_OPERATIONS_CONFIG[currentOperation].zero; + text = currentOperation + ' ' + value; } if (visible) { this.$['*operationTooltip'] = text; @@ -310,7 +339,7 @@ export class EditorToolbar extends CloudImageEditorBase { this._updateInfoTooltip(); }); - this.sub('*originalUrl', (originalUrl) => { + this.sub('*originalUrl', () => { this.$['*faderEl'] && this.$['*faderEl'].deactivate(); }); @@ -342,6 +371,16 @@ export class EditorToolbar extends CloudImageEditorBase { this.$['presence.mainToolbar'] = !showSlider; }); + this.sub('*tabList', (tabList) => { + this.$['presence.tabToggles'] = tabList.length > 1; + this.$['*tabId'] = tabList[0]; + for (const tabId of ALL_TABS) { + this.$[`presence.tabToggle.${tabId}`] = tabList.includes(tabId); + const toggleEl = this.ref[`tab-toggle-${tabId}`]; + toggleEl.style.gridColumn = tabList.indexOf(tabId) + 1; + } + }); + this._updateInfoTooltip(); } } @@ -355,13 +394,13 @@ EditorToolbar.template = /* HTML */ `
-
${TABS.map(renderTabContent).join('')}
+
${ALL_TABS.map(renderTabContent).join('')}
-
+
- ${TABS.map(renderTabToggle).join('')} -
+ ${ALL_TABS.map(renderTabToggle).join('')} +
diff --git a/blocks/CloudImageEditor/src/css/common.css b/blocks/CloudImageEditor/src/css/common.css index ffcbb6ef0..0f102d835 100644 --- a/blocks/CloudImageEditor/src/css/common.css +++ b/blocks/CloudImageEditor/src/css/common.css @@ -697,6 +697,18 @@ lr-editor-toolbar > .toolbar-container > .sub-toolbar > .tab-content-row > .tab- pointer-events: none; } +lr-editor-toolbar > .toolbar-container > .sub-toolbar > .controls-row > .tab-toggles > .tab-toggle.tab-toggle--visible { + display: contents; +} + +lr-editor-toolbar > .toolbar-container > .sub-toolbar > .controls-row > .tab-toggles > .tab-toggle.tab-toggle--hidden { + display: none; +} + +lr-editor-toolbar > .toolbar-container > .sub-toolbar > .controls-row > .tab-toggles.tab-toggles--hidden { + display: none; +} + lr-editor-toolbar > .toolbar-container > .sub-toolbar > .tab-content-row > .tab-content .controls-list_align { display: grid; grid-template-areas: '. inner .'; diff --git a/blocks/CloudImageEditor/src/css/icons.css b/blocks/CloudImageEditor/src/css/icons.css index 2faea831e..1647abcab 100644 --- a/blocks/CloudImageEditor/src/css/icons.css +++ b/blocks/CloudImageEditor/src/css/icons.css @@ -30,7 +30,7 @@ --icon-sad: 'M2 17c4.41828-4 11.5817-4 16 0M16.5 5c0 .55228-.4477 1-1 1s-1-.44772-1-1 .4477-1 1-1 1 .44772 1 1zm-11 0c0 .55228-.44772 1-1 1s-1-.44772-1-1 .44772-1 1-1 1 .44772 1 1z'; --icon-closeMax: 'M3 3l14 14m0-14L3 17'; --icon-crop: 'M20 14H7.00513C6.45001 14 6 13.55 6 12.9949V0M0 6h13.0667c.5154 0 .9333.41787.9333.93333V20M14.5.399902L13 1.9999l1.5 1.6M13 2h2c1.6569 0 3 1.34315 3 3v2M5.5 19.5999l1.5-1.6-1.5-1.6M7 18H5c-1.65685 0-3-1.3431-3-3v-2'; - --icon-sliders: 'M8 10h11M1 10h4M1 4.5h11m3 0h4m-18 11h11m3 0h4M12 4.5a1.5 1.5 0 103 0 1.5 1.5 0 10-3 0M5 10a1.5 1.5 0 103 0 1.5 1.5 0 10-3 0M12 15.5a1.5 1.5 0 103 0 1.5 1.5 0 10-3 0'; + --icon-tuning: 'M8 10h11M1 10h4M1 4.5h11m3 0h4m-18 11h11m3 0h4M12 4.5a1.5 1.5 0 103 0 1.5 1.5 0 10-3 0M5 10a1.5 1.5 0 103 0 1.5 1.5 0 10-3 0M12 15.5a1.5 1.5 0 103 0 1.5 1.5 0 10-3 0'; --icon-filters: 'M4.5 6.5a5.5 5.5 0 1011 0 5.5 5.5 0 10-11 0m-3.5 6a5.5 5.5 0 1011 0 5.5 5.5 0 10-11 0m7 0a5.5 5.5 0 1011 0 5.5 5.5 0 10-11 0'; --icon-done: 'M1 10.6316l5.68421 5.6842L19 4'; --icon-original: 'M0 40L40-.00000133'; diff --git a/blocks/CloudImageEditor/src/lib/parseTabs.js b/blocks/CloudImageEditor/src/lib/parseTabs.js new file mode 100644 index 000000000..0b2d9f0eb --- /dev/null +++ b/blocks/CloudImageEditor/src/lib/parseTabs.js @@ -0,0 +1,11 @@ +// @ts-check + +import { deserealizeCsv } from '../../../utils/comma-separated.js'; +import { ALL_TABS } from '../toolbar-constants.js'; + +/** @param {string} tabs */ +export const parseTabs = (tabs) => { + if (!tabs) return ALL_TABS; + const tabList = deserealizeCsv(tabs).filter((tab) => ALL_TABS.includes(tab)); + return tabList; +}; diff --git a/blocks/CloudImageEditor/src/state.js b/blocks/CloudImageEditor/src/state.js index 1a029ab55..63e443943 100644 --- a/blocks/CloudImageEditor/src/state.js +++ b/blocks/CloudImageEditor/src/state.js @@ -2,7 +2,9 @@ import { createCdnUrl, createCdnUrlModifiers } from '../../../utils/cdn-utils.js'; import { TRANSPARENT_PIXEL_SRC } from '../../../utils/transparentPixelSrc.js'; +import { serializeCsv } from '../../utils/comma-separated.js'; import { transformationsToOperations } from './lib/transformationUtils.js'; +import { ALL_TABS } from './toolbar-constants.js'; /** @param {import('./CloudImageEditorBlock.js').CloudImageEditorBlock} fnCtx */ export function initState(fnCtx) { @@ -18,6 +20,7 @@ export function initState(fnCtx) { '*editorTransformations': {}, /** @type {import('./types.js').CropPresetList} */ '*cropPresetList': [], + '*tabList': ALL_TABS, entry: null, extension: null, @@ -33,6 +36,7 @@ export function initState(fnCtx) { uuid: null, cdnUrl: null, cropPreset: '', + tabs: serializeCsv(ALL_TABS), 'presence.networkProblems': false, 'presence.modalCaption': true, diff --git a/blocks/CloudImageEditor/src/toolbar-constants.js b/blocks/CloudImageEditor/src/toolbar-constants.js index 39cd1d74a..fcc94dd50 100644 --- a/blocks/CloudImageEditor/src/toolbar-constants.js +++ b/blocks/CloudImageEditor/src/toolbar-constants.js @@ -1,12 +1,12 @@ +// @ts-check import { OPERATIONS_ZEROS } from './lib/transformationUtils.js'; -/** @type {{ CROP: 'crop'; SLIDERS: 'sliders'; FILTERS: 'filters' }} */ -export const TabId = { +export const TabId = Object.freeze({ CROP: 'crop', - SLIDERS: 'sliders', + TUNING: 'tuning', FILTERS: 'filters', -}; -export const TABS = [TabId.CROP, TabId.SLIDERS, TabId.FILTERS]; +}); +export const ALL_TABS = [TabId.CROP, TabId.TUNING, TabId.FILTERS]; export const ALL_COLOR_OPERATIONS = [ 'brightness', @@ -65,7 +65,7 @@ export const ALL_FILTERS = [ export const ALL_CROP_OPERATIONS = ['rotate', 'mirror', 'flip']; /** KeypointsNumber is the number of keypoints loaded from each side of zero, not total number */ -export const COLOR_OPERATIONS_CONFIG = { +export const COLOR_OPERATIONS_CONFIG = Object.freeze({ brightness: { zero: OPERATIONS_ZEROS.brightness, range: [-100, 100], @@ -111,4 +111,4 @@ export const COLOR_OPERATIONS_CONFIG = { range: [0, 100], keypointsNumber: 1, }, -}; +}); diff --git a/blocks/CloudImageEditorActivity/CloudImageEditorActivity.js b/blocks/CloudImageEditorActivity/CloudImageEditorActivity.js index 16bc8c518..14bf0433b 100644 --- a/blocks/CloudImageEditorActivity/CloudImageEditorActivity.js +++ b/blocks/CloudImageEditorActivity/CloudImageEditorActivity.js @@ -50,6 +50,7 @@ export class CloudImageEditorActivity extends UploaderBlock { const instance = new CloudImageEditorBlock(); const cdnUrl = this.$.cdnUrl; const cropPreset = this.cfg.cropPreset; + const tabs = this.cfg.cloudImageEditorTabs; instance.setAttribute('ctx-name', this.ctxName); instance.setAttribute('cdn-url', cdnUrl); @@ -57,6 +58,9 @@ export class CloudImageEditorActivity extends UploaderBlock { if (cropPreset) { instance.setAttribute('crop-preset', cropPreset); } + if (tabs) { + instance.setAttribute('tabs', tabs); + } instance.addEventListener('apply', (result) => this.handleApply(result)); instance.addEventListener('cancel', () => this.handleCancel()); diff --git a/blocks/Config/initialConfig.js b/blocks/Config/initialConfig.js index 3ccb80dee..1631d956c 100644 --- a/blocks/Config/initialConfig.js +++ b/blocks/Config/initialConfig.js @@ -1,11 +1,14 @@ // @ts-check +import { ALL_TABS } from '../CloudImageEditor/src/toolbar-constants.js'; +import { serializeCsv } from '../utils/comma-separated.js'; + export const DEFAULT_CDN_CNAME = 'https://ucarecdn.com'; export const DEFAULT_BASE_URL = 'https://upload.uploadcare.com'; export const DEFAULT_SOCIAN_BASE_URL = 'https://social.uploadcare.com'; /** @type {import('../../types/exported').ConfigType} */ -export const initialConfig = Object.freeze({ +export const initialConfig = { pubkey: '', multiple: true, multipleMin: 0, @@ -18,6 +21,7 @@ export const initialConfig = Object.freeze({ store: 'auto', cameraMirror: false, sourceList: 'local, url, camera, dropbox, gdrive', + cloudImageEditorTabs: serializeCsv(ALL_TABS), maxLocalFileSizeBytes: 0, thumbSize: 76, showEmptyList: false, @@ -51,4 +55,4 @@ export const initialConfig = Object.freeze({ userAgentIntegration: '', metadata: null, -}); +}; diff --git a/blocks/Config/normalizeConfigValue.js b/blocks/Config/normalizeConfigValue.js index c84a9bcc5..d91540b30 100644 --- a/blocks/Config/normalizeConfigValue.js +++ b/blocks/Config/normalizeConfigValue.js @@ -41,6 +41,7 @@ const mapping = { showEmptyList: asBoolean, useLocalImageEditor: asBoolean, useCloudImageEditor: asBoolean, + cloudImageEditorTabs: asString, removeCopyright: asBoolean, cropPreset: asString, diff --git a/blocks/test/cloud-image-editor.htm b/blocks/test/cloud-image-editor.htm index 99200bc0a..223a7edeb 100644 --- a/blocks/test/cloud-image-editor.htm +++ b/blocks/test/cloud-image-editor.htm @@ -27,5 +27,5 @@ - + diff --git a/blocks/test/raw-regular.htm b/blocks/test/raw-regular.htm index dd3dd5d90..12057f945 100644 --- a/blocks/test/raw-regular.htm +++ b/blocks/test/raw-regular.htm @@ -30,5 +30,5 @@ - + \ No newline at end of file diff --git a/blocks/utils/comma-separated.js b/blocks/utils/comma-separated.js new file mode 100644 index 000000000..650468026 --- /dev/null +++ b/blocks/utils/comma-separated.js @@ -0,0 +1,19 @@ +// @ts-check + +/** @param {string} value */ +export const deserealizeCsv = (value) => { + if (!value) { + return []; + } + + return value.split(',').map((item) => item.trim()); +}; + +/** @param {unknown[]} value */ +export const serializeCsv = (value) => { + if (!value) { + return ''; + } + + return value.join(','); +}; diff --git a/types/exported.d.ts b/types/exported.d.ts index 64107e4a6..ffb34c76f 100644 --- a/types/exported.d.ts +++ b/types/exported.d.ts @@ -17,6 +17,7 @@ export type ConfigType = { showEmptyList: boolean; useLocalImageEditor: boolean; useCloudImageEditor: boolean; + cloudImageEditorTabs: string; removeCopyright: boolean; cropPreset: string; modalScrollLock: boolean;