diff --git a/.eslintignore b/.eslintignore index 8d486aab7c..c16c049c43 100644 --- a/.eslintignore +++ b/.eslintignore @@ -5,6 +5,8 @@ scripts/extensions/*/node_modules scripts/extensions/**/dist end-to-end-testing-helpers/dist end-to-end-testing-helpers/node_modules +superdesk-common/node_modules +superdesk-common/dist e2e/client/node_modules e2e/client/dist e2e/client/**/*.js diff --git a/docs/item-states.dot b/docs/item-states.dot new file mode 100644 index 0000000000..c30aa355f7 --- /dev/null +++ b/docs/item-states.dot @@ -0,0 +1,36 @@ +digraph { + node [ + shape=circle, + fixedsize=true, + width=1.6, + ] + + workflow [label="\nIN_PROGRESS \n / SUBMITTED \n / FETCHED \n / ROUTED"] + published [label="PUBLISHED \n / CORRECTED"] + + "INGESTED" -> workflow [label="fetch"] + "INGESTED" -> workflow [label="route"] + + workflow -> workflow [label=" send to -> SUBMITTED\n edit -> IN_PROGRESS"] + + "DRAFT" -> workflow + + workflow -> "SPIKED" + "SPIKED" -> workflow + + workflow -> published + + workflow -> "SCHEDULED" + "SCHEDULED" -> workflow + + + "SCHEDULED" -> published + + published -> "UNPUBLISHED" + "UNPUBLISHED" -> workflow + + published -> "KILLED" + published -> "RECALLED" + + published -> published [label=" correct -> CORRECTED"] +} \ No newline at end of file diff --git a/e2e/client/package.json b/e2e/client/package.json index 43eaccc62c..f2684e333c 100644 --- a/e2e/client/package.json +++ b/e2e/client/package.json @@ -16,9 +16,10 @@ }, "scripts": { "build": "npx @superdesk/build-tools build-root-repo ./", + "server": "grunt server", "start-client-server": "http-server dist -p 9000 -s &", "stop-client-server": "fuser -k 9000/tcp", - "protractor": "node run-end-to-end-tests.js", + "protractor": "protractor protractor.conf.js", "specs--compile": "find ./specs/ -name '*.js' -type f -delete && tsc -p ./specs", "specs--watch": "find ./specs/ -name '*.js' -type f -delete && tsc -p ./specs -w", "start-test-server": "cd ../server && docker-compose build && docker-compose up -d", diff --git a/e2e/client/run-end-to-end-tests.js b/e2e/client/run-end-to-end-tests.js deleted file mode 100644 index f0045a7da1..0000000000 --- a/e2e/client/run-end-to-end-tests.js +++ /dev/null @@ -1,46 +0,0 @@ -const execSync = require('child_process').execSync; - -function ensurePackageInstalled() { - return new Promise((resolve, reject) => { - try { - require.resolve('webdriver-manager'); - resolve(); - } catch (_) { - reject('Package "webdriver-manager" was not found. Run `yarn install` to install all packages.'); - } - }); -} - -function installWebdriverDriver() { - return new Promise((resolve, reject) => { - try { - require.resolve('webdriver-manager/selenium/update-config.json'); - resolve(); - } catch (_) { - // driver not installed, installing: - - const version = execSync('$CHROME_BIN --product-version').toString(); - - if (version == null) { - return reject('To launch the test server a chrome based browser has to be installed and CHROME_BIN environment variable set.'); - } - - console.info('Installing webdriver...', version); - execSync(`npx webdriver-manager update --gecko false --standalone false --versions.chrome=${version}`); - - resolve(); - } - }); -} - -ensurePackageInstalled() - .then(installWebdriverDriver) - .then(() => { - const argumentsToForward = process.argv.slice(2).join(' '); - - execSync(`npx protractor protractor.conf.js ${argumentsToForward}`, {stdio: 'inherit'}); - }) - .catch((e) => { - console.error(e); - process.exitCode = 1; - }); \ No newline at end of file diff --git a/e2e/client/specs/_unstable-tests.txt b/e2e/client/specs/_unstable-tests.txt index 8ff5c94c84..1a855b14f8 100644 --- a/e2e/client/specs/_unstable-tests.txt +++ b/e2e/client/specs/_unstable-tests.txt @@ -3,3 +3,4 @@ monitoring can display the item in Desk Output when it's published in a producti authoring related item widget can open published item send warns that there are spelling mistakes master_desk show content view - edit item +can save and use language preferences diff --git a/e2e/client/specs/archived_spec.ts b/e2e/client/specs/archived_spec.ts index 9fb56996eb..1e228eca37 100644 --- a/e2e/client/specs/archived_spec.ts +++ b/e2e/client/specs/archived_spec.ts @@ -3,6 +3,7 @@ import {globalSearch} from './helpers/search'; import {authoring} from './helpers/authoring'; import {content} from './helpers/content'; import {monitoring} from './helpers/monitoring'; +import {element, by} from 'protractor'; describe('archived', () => { beforeEach(() => { @@ -31,7 +32,7 @@ describe('archived', () => { expect(authoring.edit_correct_button.isDisplayed()).toBe(false); expect(authoring.edit_kill_button.isDisplayed()).toBe(false); expect(authoring.edit_takedown_button.isDisplayed()).toBe(false); - expect(authoring.navbarMenuBtn.isPresent()).toBe(false); + expect(element(by.css('[data-test-id="content-create"]')).isPresent()).toBe(false); expect(authoring.sendToButton.isDisplayed()).toBe(false); authoring.showInfo(); expect(authoring.isPublishedState()).toBe(true); diff --git a/e2e/client/specs/authoring_spec.ts b/e2e/client/specs/authoring_spec.ts index da0b824ca2..83dd4f3720 100644 --- a/e2e/client/specs/authoring_spec.ts +++ b/e2e/client/specs/authoring_spec.ts @@ -50,61 +50,7 @@ describe('authoring', () => { monitoring.openMonitoring(); }); - it('add an embed and respect the order', () => { - // try with same block content - monitoring.actionOnItem('Edit', 2, 0); - authoring.cleanBodyHtmlElement(); - authoring.writeText('line\n'); - authoring.addEmbed('embed'); - var thirdBlockContext = element(by.model('item.body_html')).all(by.repeater('block in vm.blocks')).get(2); - - thirdBlockContext.all(by.css('.editor-type-html')).first().sendKeys('line\n'); - authoring.addEmbed('embed', thirdBlockContext); - authoring.blockContains(0, 'line'); - authoring.blockContains(1, 'embed'); - authoring.blockContains(2, 'line'); - authoring.blockContains(3, 'embed'); - authoring.close(); - authoring.ignore(); - // with different block content - monitoring.actionOnItem('Edit', 2, 0); - authoring.cleanBodyHtmlElement(); - function generateLines(from, to) { - var lines = ''; - - for (var j = from; j < to; j++) { - lines += 'line ' + j + '\n'; - } - return lines; - } - var body1 = generateLines(0, 8); - var body2 = generateLines(8, 15); - var body3 = generateLines(15, 20); - - authoring.writeText(body1 + body2 + body3.trim()); - authoring.writeText( - protractor.Key.HOME + - protractor.Key.UP.repeat(4) + - protractor.Key.ENTER + - protractor.Key.UP, - ); - - authoring.addEmbed('Embed at position 15'); - authoring.blockContains(0, (body1 + body2).replace(/\n$/, '')); - authoring.blockContains(2, body3.replace(/\n$/, '')); - authoring.writeText( - protractor.Key.UP.repeat(7) + - protractor.Key.ENTER, - ); - authoring.addEmbed('Embed at position 8'); - authoring.blockContains(0, body1.replace(/\n$/, '')); - authoring.blockContains(2, body2.replace(/\n$/, '')); - authoring.blockContains(4, body3.replace(/\n$/, '')); - }); - - it('authoring operations', () => { - // undo and redo operations by using CTRL+Z and CTRL+y ... - // ... from a new item + it('can undo and redo', () => { authoring.createTextItem(); browser.sleep(1000); authoring.writeText('to be undone'); @@ -114,40 +60,13 @@ describe('authoring', () => { expect(authoring.getBodyText()).toBe(''); ctrlKey('y'); expect(authoring.getBodyText()).toBe('to be undone'); - authoring.writeText(protractor.Key.HOME + protractor.Key.ENTER + protractor.Key.UP); - authoring.addEmbed('Embed'); - authoring.blockContains(1, 'Embed'); - authoring.blockContains(2, 'to be undone'); - commandKey('z'); - authoring.blockContains(0, 'to be undone'); - commandKey('y'); - authoring.blockContains(1, 'Embed'); - authoring.blockContains(2, 'to be undone'); - - authoring.cutBlock(1); - authoring.blockContains(0, 'to be undone'); - ctrlKey('z'); - authoring.blockContains(1, 'Embed'); - authoring.blockContains(2, 'to be undone'); - authoring.close(); - authoring.ignore(); - // ... from an existing item - expect(monitoring.getTextItem(2, 0)).toBe('item5'); - monitoring.actionOnItem('Edit', 2, 0); - expect(authoring.getBodyText()).toBe('item5 text'); - authoring.writeText(' Two'); - expect(authoring.getBodyText()).toBe('item5 text Two'); - authoring.writeText(' Words'); - expect(authoring.getBodyText()).toBe('item5 text Two Words'); - ctrlKey('z'); - expect(authoring.getBodyText()).toBe('item5 text Two'); - ctrlKey('y'); - expect(authoring.getBodyText()).toBe('item5 text Two Words'); - authoring.save(); - authoring.close(); + }); + it('authoring operations', () => { // allows to create a new empty package - monitoring.createItemAction('create_package'); + el(['content-create']).click(); + el(['content-create-dropdown', 'create-package']).click(); + expect(element(by.className('packaging-screen')).isDisplayed()).toBe(true); authoring.close(); @@ -335,7 +254,13 @@ describe('authoring', () => { authoring.writeText('z'); element(by.cssContainingText('label', 'Dateline')).click(); ctrlShiftKey('s'); - browser.wait(() => element(by.buttonText('Save')).getAttribute('disabled'), 500); + + browser.wait(ECE.attributeEquals( + element(by.buttonText('Save')), + 'disabled', + 'true', + )); + authoring.close(); monitoring.actionOnItem('Edit', 2, 0); browser.sleep(300); @@ -405,39 +330,31 @@ describe('authoring', () => { it('toggle auto spellcheck and hold changes', () => { monitoring.actionOnItem('Edit', 2, 1); - expect(element(by.model('spellcheckMenu.isAuto')).getAttribute('checked')).toBeTruthy(); + + browser.wait(ECE.attributeEquals( + element(by.model('spellcheckMenu.isAuto')), + 'checked', + 'true', + )); + authoring.toggleAutoSpellCheck(); - expect(element(by.model('spellcheckMenu.isAuto')).getAttribute('checked')).toBeFalsy(); - authoring.close(); - monitoring.actionOnItem('Edit', 2, 2); - expect(element(by.model('spellcheckMenu.isAuto')).getAttribute('checked')).toBeFalsy(); - }); - it('spellcheck hilite sentence word for capitalization and ignore the word after abbreviations', () => { - nav('/settings/dictionaries'); - dictionaries.edit('Test 1'); - expect(dictionaries.getWordsCount()).toBe(0); - dictionaries.search('abbrev.'); - dictionaries.saveWord(); - dictionaries.search('abbrev'); - dictionaries.saveWord(); - expect(dictionaries.getWordsCount()).toBe(2); - dictionaries.save(); - browser.sleep(200); + browser.wait(ECE.attributeEquals( + element(by.model('spellcheckMenu.isAuto')), + 'checked', + null, + )); - monitoring.openMonitoring(); + authoring.close(); - authoring.createTextItem(); - authoring.writeText('some is a sentence word, but words come after an abbrev. few are not'); - browser.sleep(200); - expect(authoring.getBodyInnerHtml()).toContain('some'); - expect(authoring.getBodyInnerHtml()).not.toContain('few'); - expect(authoring.getBodyInnerHtml()).toContain('few'); - }); + monitoring.actionOnItem('Edit', 2, 2); + browser.wait(ECE.attributeEquals( + element(by.model('spellcheckMenu.isAuto')), + 'checked', + null, + )); + }); it('related item widget', () => { monitoring.actionOnItem('Edit', 2, 1); authoring.writeText('something'); @@ -585,8 +502,15 @@ describe('authoring', () => { expect(authoring.getInnerDropdownItemVersions(1).count()).toBe(2); authoring.openItemVersionInBoard(1, 0); expect(authoring.getInnerDropdownItemVersions(0).count()).toBe(1); - expect(authoring.getHtmlArticleHeadlineOfBoard(0)).toContain( - 'item5 updated newly', + expect( + authoring.getHtmlArticleHeadlineOfBoard(0).then((text) => { + return text + .replace(/ data-text="true"/g, '') + .replace(/ data-offset-key=".+?"/g, ''); + }), + ).toContain( + 'item5 updated' + + ' newly', ); expect(authoring.getArticleHeadlineOfBoard(1)).toEqual('item5 updated'); }); @@ -670,22 +594,6 @@ describe('authoring', () => { expect(authoring.send_kill_button.isDisplayed()).toBeTruthy(); }); - it('after undo/redo save last version', () => { - monitoring.actionOnItem('Edit', 2, 0); - authoring.cleanBodyHtmlElement(); - browser.sleep(2000); - authoring.writeText('one\ntwo\nthree'); - browser.sleep(2000); // wait for autosave - authoring.backspaceBodyHtml(5); - browser.sleep(2000); - ctrlKey('z'); - browser.sleep(1000); - authoring.save(); - authoring.close(); - monitoring.actionOnItem('Edit', 2, 0); - expect(authoring.getBodyText()).toBe('one\ntwo\nthree'); - }); - it('can minimize story while a correction and kill is being written', () => { workspace.selectDesk('Politic Desk'); expect(monitoring.getTextItem(3, 2)).toBe('item6'); diff --git a/e2e/client/specs/content_profile_spec.ts b/e2e/client/specs/content_profile_spec.ts index 800fd1822c..f52735ada0 100644 --- a/e2e/client/specs/content_profile_spec.ts +++ b/e2e/client/specs/content_profile_spec.ts @@ -7,20 +7,19 @@ import {workspace} from './helpers/workspace'; import {authoring} from './helpers/authoring'; import {metadata} from './helpers/metadata'; import {assertToastMsg} from './helpers/utils'; +import {ECE} from '@superdesk/end-to-end-testing-helpers'; describe('Content profiles', () => { it('creates corresponding template', () => { // create a new content profile contentProfiles.openContentProfileSettings(); - contentProfiles.addNew('Simple'); + contentProfiles.addNew('Simple', 'text'); contentProfiles.toggleEnable(); - element(by.buttonText('Content fields')).click(); - contentProfiles.disableField('Abstract'); + contentProfiles.editContentFields(); contentProfiles.update(); templates.openTemplatesSettings(); expect(templates.getListCount()).toBeGreaterThan(2); templates.edit('Simple'); - expect(authoring.getAbstractFieldCount()).toEqual(0); expect(templates.getContentProfile()).toEqual('Simple'); templates.cancel(); @@ -38,7 +37,6 @@ describe('Content profiles', () => { templates.openTemplatesSettings(); expect(templates.getListCount()).toBeGreaterThan(2); templates.edit('Simple'); - expect(authoring.getAbstractFieldCount()).toEqual(1); expect(templates.getContentProfile()).toEqual(''); templates.cancel(); }); @@ -46,11 +44,8 @@ describe('Content profiles', () => { it('displays defined fields in authoring', () => { // create a new content profile contentProfiles.openContentProfileSettings(); - contentProfiles.addNew('Simple'); + contentProfiles.addNew('Simple', 'text'); contentProfiles.toggleEnable(); - element(by.buttonText('Content fields')).click(); - contentProfiles.disableField('Abstract'); - element(by.buttonText('Header fields')).click(); contentProfiles.setRequired('Ed. Note'); contentProfiles.update(); templates.openTemplatesSettings(); @@ -62,7 +57,6 @@ describe('Content profiles', () => { monitoring.openMonitoring(); workspace.selectDesk('Sports Desk'); authoring.createTextItemFromTemplate('simple'); - expect(authoring.getAbstractFieldCount()).toEqual(0); // publish of the required field will fail authoring.setHeaderSluglineText('Story1 slugline'); @@ -90,16 +84,16 @@ describe('Content profiles', () => { expect(metadata.items().count()).toBe(1); contentProfiles.openContentProfileSettings(); - contentProfiles.addNew('Simple'); + contentProfiles.addNew('Simple', 'text'); - element(by.buttonText('Content fields')).click(); + contentProfiles.editContentFields(); - const btns = element.all(by.partialButtonText(FIELD_LABEL)); + const buttons = element.all(by.partialButtonText(FIELD_LABEL)); - expect(btns.filter((elem) => elem.isDisplayed()).count()).toBe(0); + browser.wait(ECE.hasElementCount(buttons, 0)); contentProfiles.openAddFieldDropdown(); - expect(btns.filter((elem) => elem.isDisplayed()).count()).toBe(1); + browser.wait(ECE.hasElementCount(buttons, 1)); }); }); diff --git a/e2e/client/specs/content_spec.ts b/e2e/client/specs/content_spec.ts index 7b8a5d255f..91c14a0026 100644 --- a/e2e/client/specs/content_spec.ts +++ b/e2e/client/specs/content_spec.ts @@ -7,6 +7,7 @@ import {content} from './helpers/content'; import {authoring} from './helpers/authoring'; import {multiAction} from './helpers/actions'; import {ECE, el} from '@superdesk/end-to-end-testing-helpers'; +import {TreeSelectDriver} from './helpers/tree-select-driver'; describe('content', () => { var body = element(by.tagName('body')); @@ -33,8 +34,8 @@ describe('content', () => { var embargoTime = (now.getHours() < 10 ? '0' + now.getHours() : now.getHours()) + ':' + (now.getMinutes() < 10 ? '0' + now.getMinutes() : now.getMinutes()); - element(by.model('item.embargo_date')).element(by.tagName('input')).sendKeys(embargoDate); - element(by.model('item.embargo_time')).element(by.tagName('input')).sendKeys(embargoTime); + el(['authoring', 'interactive-actions-panel', 'embargo', 'date-input']).sendKeys(embargoDate); + el(['authoring', 'interactive-actions-panel', 'embargo', 'time-input']).sendKeys(embargoTime); } it('can navigate with keyboard', () => { @@ -141,7 +142,7 @@ describe('content', () => { browser.sleep(100); - multiAction('Multiedit'); + multiAction('Multi-edit'); expect(browser.getCurrentUrl()).toMatch(/multiedit$/); expect(element.all(by.repeater('board in boards')).count()).toBe(2); }); @@ -150,8 +151,7 @@ describe('content', () => { workspace.switchToDesk('SPORTS DESK'); content.setListView(); - element(by.className('sd-create-btn')).click(); - element(by.id('create_text_article')).click(); + authoring.createTextItem(); authoring.writeText('Words'); authoring.save(); @@ -164,8 +164,8 @@ describe('content', () => { workspace.switchToDesk('SPORTS DESK'); content.setListView(); - element(by.className('sd-create-btn')).click(); - element(by.id('create_package')).click(); + el(['content-create']).click(); + el(['content-create-dropdown', 'create-package']).click(); element.all(by.model('item.headline')).first().sendKeys('Empty Package'); authoring.save(); @@ -220,10 +220,10 @@ describe('content', () => { it('can display embargo in metadata when set', () => { workspace.editItem('item3', 'SPORTS'); - authoring.sendToButton.click(); + + el(['open-send-publish-pane']).click(); setEmbargo(); - browser.sleep(100); authoring.closeSendAndPublish(); @@ -238,37 +238,46 @@ describe('content', () => { content.closePreview(); }); - it('can enable/disable send based on embargo', () => { + it('can set embargo and send', () => { // Initial steps before proceeding, to get initial state of send buttons. workspace.editItem('item3', 'SPORTS'); authoring.sendTo('Sports Desk', 'Incoming Stage'); authoring.confirmSendTo(); workspace.editItem('item3', 'SPORTS'); - authoring.sendToButton.click().then(() => { - // Initial State - expect(authoring.sendBtn.isEnabled()).toBe(false); - }); - var sidebar = element.all(by.css('.side-panel')).last(), - dropdown = sidebar.element(by.css('.dropdown--boxed .dropdown__toggle')); + el(['open-send-publish-pane']).click(); + + el(['authoring', 'interactive-actions-panel', 'tabs'], by.buttonText('Send to')).click(); + + const sendToButton = el(['authoring', 'interactive-actions-panel', 'send']); - dropdown.waitReady(); - dropdown.click(); - sidebar.element(by.buttonText('Sports Desk')).click(); + browser.wait(ECE.visibilityOf(sendToButton)); + + new TreeSelectDriver( + el(['interactive-actions-panel', 'destination-select']), + ).setValue('Sports Desk'); + + const stage = 'two'; // State after selecting different Stage in the same desk - sidebar.element(by.buttonText('two')).click(); - expect(authoring.sendBtn.isEnabled()).toBe(true); + el( + ['interactive-actions-panel', 'stage-select'], + by.cssContainingText('[data-test-id="item"]', stage), + ).click(); + + expect(sendToButton.isEnabled()).toBe(true); // State after setting Embargo setEmbargo(); browser.sleep(100); - expect(authoring.sendBtn.isEnabled()).toBe(true); + expect(sendToButton.isEnabled()).toBe(true); // State after changing Desk - dropdown.click(); - sidebar.element(by.buttonText('Politic Desk')).click(); - expect(authoring.sendBtn.isEnabled()).toBe(true); + new TreeSelectDriver( + el(['interactive-actions-panel', 'destination-select']), + ).setValue('Politic Desk'); + + expect(sendToButton.isEnabled()).toBe(true); }); }); diff --git a/e2e/client/specs/desks_spec.ts b/e2e/client/specs/desks_spec.ts index 7a95acfe10..8cea53cbe3 100644 --- a/e2e/client/specs/desks_spec.ts +++ b/e2e/client/specs/desks_spec.ts @@ -207,7 +207,7 @@ describe('desks', () => { contentProfiles.openContentProfileSettings(); contentProfiles.edit('testing'); contentProfiles.setRequired('Subject'); - element(by.buttonText('Content fields')).click(); + contentProfiles.editContentFields(); contentProfiles.setRequired('Body HTML'); contentProfiles.update(); diff --git a/e2e/client/specs/fetch_spec.ts b/e2e/client/specs/fetch_spec.ts index 7c909be33f..8d0650f671 100644 --- a/e2e/client/specs/fetch_spec.ts +++ b/e2e/client/specs/fetch_spec.ts @@ -5,7 +5,7 @@ import {content} from './helpers/content'; import {authoring} from './helpers/authoring'; import {desks} from './helpers/desks'; import {multiAction} from './helpers/actions'; -import {ECE, els} from '@superdesk/end-to-end-testing-helpers'; +import {ECE, el, els, s} from '@superdesk/end-to-end-testing-helpers'; describe('fetch', () => { beforeEach(() => { @@ -31,7 +31,7 @@ describe('fetch', () => { it('can fetch as', () => { workspace.openIngest(); content.actionOnItem('Fetch To', 0); - content.send(); + el(['interactive-actions-panel', 'fetch']).click(); workspace.openContent(); expect(content.count()).toBe(3); }); @@ -46,9 +46,9 @@ describe('fetch', () => { workspace.openIngest(); content.actionOnItem('Fetch To', 0); - var btnFetchAndOpen = element(by.css('[ng-disabled="disableFetchAndOpenButton()"]')); + var btnFetchAndOpen = element(s(['interactive-actions-panel', 'fetch-and-open'])); - expect(btnFetchAndOpen.getAttribute('disabled')).toBeFalsy(); + expect(btnFetchAndOpen.isEnabled()).toBe(true); // Adding a new desk with no member, which serves as a non-member desk when selected desks.openDesksSettings(); @@ -59,21 +59,32 @@ describe('fetch', () => { desks.setDeskType('authoring'); desks.setDeskDefaultContentTemplate('testing'); desks.setDeskDefaultContentProfile('testing'); - desks.actionDoneOnGeneralTab(); + desks.actionSaveAndContinueOnGeneralTab(); // save desk and continue to Stages tab + + desks.editStage('Working Stage'); + desks.toggleGlobalReadFlag(); // turn OFF Global Read + desks.saveEditedStage(); + + desks.editStage('Incoming Stage'); + desks.toggleGlobalReadFlag(); // turn OFF Global Read + desks.saveEditedStage(); + + desks.actionDoneOnStagesTab(); workspace.openIngest(); content.actionOnItem('Fetch To', 0); authoring.selectDeskforSendTo('Test Desk'); - expect(btnFetchAndOpen.getAttribute('disabled')).toBeTruthy(); + + expect(btnFetchAndOpen.isEnabled()).toBe(false); }); it('can hide stage with global read OFF if selected desk as a non-member', () => { workspace.openIngest(); content.actionOnItem('Fetch To', 0); - var btnFetchAndOpen = element(by.css('[ng-disabled="disableFetchAndOpenButton()"]')); + var btnFetchAndOpen = element(s(['interactive-actions-panel', 'fetch-and-open'])); - expect(btnFetchAndOpen.getAttribute('disabled')).toBeFalsy(); + expect(btnFetchAndOpen.isEnabled()).toBe(true); // Adding a new desk with no member, which serves as a non-member desk when selected desks.openDesksSettings(); @@ -106,11 +117,12 @@ describe('fetch', () => { content.actionOnItem('Fetch To', 0); authoring.selectDeskforSendTo('Test Desk'); - var sidebar = element.all(by.css('.side-panel')).last(); - - expect(sidebar.element(by.buttonText('Working Stage')).isPresent()).toBeTruthy(); - expect(sidebar.element(by.buttonText('Test Stage')).isPresent()).toBeFalsy(); - expect(btnFetchAndOpen.getAttribute('disabled')).toBeTruthy(); + expect( + element(s(['interactive-actions-panel', 'stage-select', 'item'], 'Working Stage')).isPresent(), + ).toBeTruthy(); + expect( + element(s(['interactive-actions-panel', 'stage-select', 'item'], 'Test Stage')).isPresent(), + ).toBeFalsy(); }); it('can fetch multiple items', () => { @@ -127,7 +139,7 @@ describe('fetch', () => { content.selectItem(0); browser.sleep(1000); // Wait for animation multiAction('Fetch to'); - content.send(); + el(['interactive-actions-panel', 'fetch']).click(); workspace.openContent(); expect(content.count()).toBe(3); }); diff --git a/e2e/client/specs/helpers/authoring.ts b/e2e/client/specs/helpers/authoring.ts index 45ccd21939..3af555241e 100644 --- a/e2e/client/specs/helpers/authoring.ts +++ b/e2e/client/specs/helpers/authoring.ts @@ -3,10 +3,11 @@ import {element, by, browser, protractor} from 'protractor'; import {waitHidden, waitFor, click} from './utils'; import {ECE, els, el} from '@superdesk/end-to-end-testing-helpers'; +import {PLAIN_TEXT_TEMPLATE_NAME} from './constants'; +import {TreeSelectDriver} from './tree-select-driver'; class Authoring { lock: any; - publish_button: any; correct_button: any; kill_button: any; close_button: any; @@ -20,27 +21,21 @@ class Authoring { edit_correct_button: any; edit_kill_button: any; edit_takedown_button: any; - navbarMenuBtn: any; - newPlainArticleLink: any; newEmptyPackageLink: any; infoIconsBox: any; sendToButton: any; - sendAndContinueBtn: any; sendAndPublishBtn: any; - sendBtn: any; moreActionsButton: any; multieditButton: any; compareVersionsMenuItem: any; setCategoryBtn: any; getCategoryListItems: any; - sendItemContainer: any; linkToMasterButton: any; marked_for_legal: any; sms: any; anpa_category: any; subject: any; missing_link: any; - publish_panel: any; send_panel: any; fetch_panel: any; headline: any; @@ -67,9 +62,7 @@ class Authoring { ignore: () => any; savePublish: () => any; publish: (skipConfirm?: any) => void; - sendAndpublish: (desk: any, skipConfirm?: any) => void; closeSendAndPublish: () => any; - publishFrom: (desk: any) => void; schedule: (skipConfirm?: any) => void; correct: () => any; save: () => any; @@ -180,10 +173,10 @@ class Authoring { openCompareVersionsInnerDropdown: (index: any) => void; getInnerDropdownItemVersions: (index: any) => any; openItemVersionInBoard: (board: any, index: any) => void; + createPlainTextArticle: () => void; constructor() { this.lock = element(by.css('[ng-click="lock()"]')); - this.publish_button = element(by.buttonText('publish')); this.correct_button = element(by.buttonText('correct')); this.kill_button = element(by.buttonText('kill')); this.close_button = element(by.buttonText('Close')); @@ -198,15 +191,11 @@ class Authoring { this.edit_kill_button = element(by.css('[title="Kill"]')); this.edit_takedown_button = element(by.css('[title="Takedown"]')); - this.navbarMenuBtn = element(by.css('.dropdown__toggle.sd-create-btn')); - this.newPlainArticleLink = element(by.id('create_text_article')); this.newEmptyPackageLink = element(by.id('create_package')); this.infoIconsBox = element(by.css('.info-icons')); this.sendToButton = element(by.id('send-to-btn')); - this.sendAndContinueBtn = element(by.buttonText('send and continue')); this.sendAndPublishBtn = element(by.buttonText('publish from')); - this.sendBtn = element(by.buttonText('send')); this.moreActionsButton = element(by.id('more-actions')); @@ -219,7 +208,6 @@ class Authoring { this.getCategoryListItems = element(by.id('category-setting')) .all(el(['dropdown__item']).locator()); - this.sendItemContainer = element(by.id('send-item-container')); this.linkToMasterButton = element(by.id('preview-master')); this.marked_for_legal = element(by.model('item.flags.marked_for_legal')); this.sms = element(by.model('item.flags.marked_for_sms')); @@ -227,10 +215,9 @@ class Authoring { .all(by.css('[data-field="anpa_category"]')); this.subject = element(by.className('authoring-header__detailed')).all(by.css('[data-field="subject"]')); this.missing_link = element(by.className('missing-link')); - this.publish_panel = element(by.css('#panel-publish:not(.ng-hide)')); this.send_panel = element(by.css('#panel-send:not(.ng-hide)')); this.fetch_panel = element(by.css('#panel-fetch:not(.ng-hide)')); - this.headline = element(by.model('item.headline')).all(by.className('editor-type-html')).first(); + this.headline = element(by.css('.headline [contenteditable]')); this.send_kill_button = element(by.id('send-kill-btn')); this.send_correction_button = element(by.id('send-correction-btn')); @@ -280,8 +267,8 @@ class Authoring { var embargoDate = '09/09/' + ((new Date()).getFullYear() + 1); var embargoTime = '04:00'; - element(by.model('item.embargo_date')).element(by.tagName('input')).sendKeys(embargoDate); - element(by.model('item.embargo_time')).element(by.tagName('input')).sendKeys(embargoTime); + el(['authoring', 'interactive-actions-panel', 'embargo', 'date-input']).sendKeys(embargoDate); + el(['authoring', 'interactive-actions-panel', 'embargo', 'time-input']).sendKeys(embargoTime); }; this.confirmSendTo = function() { @@ -293,62 +280,55 @@ class Authoring { }; this.sendToSidebarOpened = function(desk, stage, _continue) { - browser.wait(ECE.elementToBeClickable(this.send_panel)); - this.send_panel.click(); + el(['interactive-actions-panel', 'tabs'], by.buttonText('Send to')).click(); - var sidebar = element.all(by.css('.side-panel')).last(), - dropdown = sidebar.element(by.css('.dropdown--boxed .dropdown__toggle')); + new TreeSelectDriver( + el(['interactive-actions-panel', 'destination-select']), + ).setValue(desk); - dropdown.waitReady(); - dropdown.click(); - sidebar.element(by.buttonText(desk)).click(); if (stage) { - sidebar.element(by.buttonText(stage)).click(); + el( + ['interactive-actions-panel', 'stage-select'], + by.cssContainingText('[data-test-id="item"]', stage), + ).click(); } if (_continue) { - this.sendAndContinueBtn.click(); + el(['interactive-actions-panel', 'send-and-open']).click(); } else { - this.sendBtn.click(); + el(['interactive-actions-panel', 'send']).click(); } }; this.duplicateTo = (desk, stage, open) => { - let duplicateButton = element(by.id('duplicate-btn')); - let duplicateAndOpenButton = element(by.id('duplicate-open-btn')); + new TreeSelectDriver( + el(['interactive-actions-panel', 'destination-select']), + ).setValue(desk); - var sidebar = element.all(by.css('.side-panel')).last(), - dropdown = sidebar.element(by.css('.dropdown--boxed .dropdown__toggle')); - - dropdown.waitReady(); - dropdown.click(); - sidebar.element(by.buttonText(desk)).click(); if (stage) { - sidebar.element(by.buttonText(stage)).click(); + el( + ['interactive-actions-panel', 'stage-select'], + by.cssContainingText('[data-test-id="item"]', stage), + ).click(); } if (open) { - duplicateAndOpenButton.click(); + el(['interactive-actions-panel', 'duplicate-and-open']).click(); } else { - duplicateButton.click(); + el(['interactive-actions-panel', 'duplicate']).click(); } }; this.selectDeskforSendTo = function(desk) { - var sidebar = element.all(by.css('.side-panel')).last(), - dropdown = element(by.css('.dropdown--boxed .dropdown__toggle')); - - dropdown.waitReady(); - dropdown.click(); - sidebar.element(by.buttonText(desk)).click(); + new TreeSelectDriver( + el(['interactive-actions-panel', 'destination-select']), + ).setValue(desk); }; this.markAction = function() { return element(by.className('svg-icon-add-to-list')).click(); }; - this.createTextItem = function() { - return element(by.className('sd-create-btn')) - .click() - .then(() => element(by.id('create_text_article')).click()); + this.createTextItem = () => { + this.createTextItemFromTemplate(PLAIN_TEXT_TEMPLATE_NAME); }; /** @@ -357,13 +337,11 @@ class Authoring { * @param {String} name */ this.createTextItemFromTemplate = (name) => { - element(by.className('sd-create-btn')).click(); - element(by.id('more_templates')).click(); - let templates = element.all(by.repeater('template in templates track by template._id')); - - templates.all(by.css('[ng-click="select({template: template})"]')) - .filter((elem) => elem.getText().then((text) => text.toUpperCase().indexOf(name.toUpperCase()) > -1)) - .click(); + el(['content-create']).click(); + el(['content-create-dropdown'], by.buttonText('More templates...')).click(); + el(['content-create-dropdown', 'search']).sendKeys(name); + el(['content-create-dropdown'], by.buttonText(name)).click(); + browser.wait(ECE.presenceOf(el(['authoring']))); }; this.close = function() { @@ -423,40 +401,8 @@ class Authoring { }; this.publish = function(skipConfirm) { - browser.wait(() => this.sendToButton.isPresent(), 1000); - this.sendToButton.click(); - - browser.wait(() => this.publish_panel.isPresent(), 3000); - - this.publish_panel.click(); - - browser.wait(() => this.publish_button.isPresent(), 3000); - - this.publish_panel.click(); - this.publish_button.click(); - - if (!skipConfirm) { - var modal = element(by.className('modal__dialog')); - - modal.isPresent().then((isPresent) => { - if (isPresent) { - modal.element(by.className('btn--primary')).click(); - } - }); - } - }; - - this.sendAndpublish = function(desk, skipConfirm) { - browser.wait(() => this.sendToButton.isPresent(), 1000); - this.sendToButton.click(); - - this.publish_panel.click(); - - browser.wait(() => this.publish_button.isPresent(), 1000); - - this.publish_panel.click(); - this.selectDeskforSendTo(desk); - this.sendAndPublishBtn.click(); + el(['authoring', 'open-send-publish-pane']).click(); + el(['authoring', 'interactive-actions-panel', 'publish']).click(); if (!skipConfirm) { var modal = element(by.className('modal__dialog')); @@ -470,17 +416,7 @@ class Authoring { }; this.closeSendAndPublish = function() { - var sidebar = element.all(by.css('.side-panel')).last(); - - return sidebar.element(by.css('[ng-click="close()"]')).click(); - }; - - this.publishFrom = function(desk) { - this.publish_panel.click(); - - browser.wait(() => this.publish_panel.isPresent(), 2000); - this.selectDeskforSendTo(desk); - this.sendAndPublishBtn.click(); + el(['authoring', 'interactive-actions-panel', 'close']).click(); }; this.schedule = function(skipConfirm) { @@ -490,14 +426,10 @@ class Authoring { var scheduleDate = '09/09/' + ((new Date()).getFullYear() + 1); var scheduleTime = '04:00'; - element(by.model('item.publish_schedule_date')).element(by.tagName('input')).sendKeys(scheduleDate); - element(by.model('item.publish_schedule_time')).element(by.tagName('input')).sendKeys(scheduleTime); + el(['authoring', 'interactive-actions-panel', 'publish-schedule', 'date-input']).sendKeys(scheduleDate); + el(['authoring', 'interactive-actions-panel', 'publish-schedule', 'time-input']).sendKeys(scheduleTime); - this.publish_panel.click(); - - browser.wait(() => this.publish_button.isPresent(), 1000); - - this.publish_button.click(); + el(['authoring', 'interactive-actions-panel', 'publish']).click(); if (!skipConfirm) { var modal = element(by.className('modal__dialog')); @@ -732,11 +664,14 @@ class Authoring { return element(by.id('subnav')); }; - var getBodyHtml = () => browser.wait(ECE.presenceOf(element(by.model('item.body_html')))) - .then(() => element(by.model('item.body_html')).all(by.className('editor-type-html')).first()); + var getBodyHtml = () => { + const elem = element(by.css('.field.body [contenteditable]')); + + return browser.wait(ECE.presenceOf(elem)).then(() => elem); + }; - var abstract = element(by.model('item.abstract')).all(by.className('editor-type-html')).first(); - var bodyFooter = element(by.id('body_footer')).all(by.className('editor-type-html')).first(); + var abstract = element(by.css('.abstract [contenteditable]')); + var bodyFooter = element(by.css('#body_footer [contenteditable]')); var packageSlugline = element.all(by.className('keyword')).last(); var byline = element(by.model('item.byline')).all(by.className('editor-type-html')).first(); @@ -843,7 +778,7 @@ class Authoring { }; this.getAbstractFieldCount = function() { - return element.all(by.model('item.abstract')).count(); + return element.all(by.css('.abstract [contenteditable]')).count(); }; this.closeHeader = function() { diff --git a/e2e/client/specs/helpers/constants.ts b/e2e/client/specs/helpers/constants.ts new file mode 100644 index 0000000000..fd2eec0ef5 --- /dev/null +++ b/e2e/client/specs/helpers/constants.ts @@ -0,0 +1 @@ +export const PLAIN_TEXT_TEMPLATE_NAME = 'plain text'; diff --git a/e2e/client/specs/helpers/content.ts b/e2e/client/specs/helpers/content.ts index 11cc465480..935ac47474 100644 --- a/e2e/client/specs/helpers/content.ts +++ b/e2e/client/specs/helpers/content.ts @@ -162,7 +162,7 @@ class Content { this.unspikeItems = function() { multiAction('Unspike'); - element(by.buttonText('send')).click(); + el(['interactive-actions-panel', 'unspike']).click(); }; this.selectSpikedList = function() { diff --git a/e2e/client/specs/helpers/content_profiles.ts b/e2e/client/specs/helpers/content_profiles.ts index 5d7a544d77..7c852b88b2 100644 --- a/e2e/client/specs/helpers/content_profiles.ts +++ b/e2e/client/specs/helpers/content_profiles.ts @@ -1,13 +1,21 @@ /* eslint-disable newline-per-chained-call */ -import {element, by, ElementFinder, browser} from 'protractor'; +import {el} from '@superdesk/end-to-end-testing-helpers'; +import {element, by, ElementFinder, browser, ExpectedConditions as EC} from 'protractor'; import {nav} from './utils'; +const clickButton = (label) => { + const button = element(by.buttonText(label)); + + browser.wait(EC.elementToBeClickable(button), 1000); + button.click(); +}; + class ContentProfiles { list: any; openContentProfileSettings: () => void; add: () => void; - addNew: (name: any) => void; + addNew: (name: string, type: 'text' | 'picture') => void; getNameElement: () => ElementFinder; save: () => void; getRow: (name: any) => any; @@ -19,6 +27,8 @@ class ContentProfiles { setRequired: (fieldName: any) => void; cancel: () => void; openAddFieldDropdown: () => void; + editContentFields: () => void; + editHeaderFields: () => void; constructor() { /** List of content profiles on settings page **/ @@ -38,9 +48,14 @@ class ContentProfiles { element(by.id('add-new-content-profile')).click(); }; - this.addNew = (name) => { + this.addNew = (name, type) => { + const modal = element(by.className('modal__body')); + const typeIcon = modal.element(by.className('icon-' + type)); + this.add(); this.getNameElement().sendKeys(name); + browser.wait(EC.elementToBeClickable(typeIcon), 1000); + typeIcon.click(); this.save(); }; @@ -129,12 +144,18 @@ class ContentProfiles { * @param {string} name of field **/ this.setRequired = function(fieldName) { - const requiredCheckbox = element(by.cssContainingText('.title', fieldName)) - .element(by.xpath('..')) - .element(by.xpath('..')) - .element(by.model('model.schema[id].required')); + const field = element(by.cssContainingText('.sd-list-item', fieldName)); + const saveButton = el(['item-view-edit--save']); + const requiredLabel = element(by.cssContainingText('.form__row', 'Required')).element(by.tagName('label')); + + browser.actions().mouseMove(field).perform(); + field.click(); - requiredCheckbox.click(); + browser.wait(EC.elementToBeClickable(requiredLabel), 1000); + requiredLabel.click(); + + browser.wait(EC.elementToBeClickable(saveButton), 1000); + saveButton.click(); }; /** @@ -148,10 +169,18 @@ class ContentProfiles { * Open first add field dropdown */ this.openAddFieldDropdown = () => { - element.all(by.className('dropdown--add-more')) - .filter((el) => el.isDisplayed()) + element.all(by.className('icon-plus-large')) + .filter((elem) => elem.isDisplayed()) .first() - .element(by.tagName('button')).click(); + .click(); + }; + + this.editContentFields = () => { + clickButton('Content fields'); + }; + + this.editHeaderFields = () => { + clickButton('Header fields'); }; } } diff --git a/e2e/client/specs/helpers/monitoring.ts b/e2e/client/specs/helpers/monitoring.ts index f20358ff7c..4cd5f76cad 100644 --- a/e2e/client/specs/helpers/monitoring.ts +++ b/e2e/client/specs/helpers/monitoring.ts @@ -17,7 +17,6 @@ class Monitoring { showSpiked: () => void; showPersonal: () => void; showSearch: () => void; - createItemAction: (action: any) => void; createFromDeskTemplate: () => any; getGroup: (group: number) => any; getGroups: () => any; @@ -60,7 +59,7 @@ class Monitoring { selectGivenItem: (item: any) => any; spikeMultipleItems: () => void; unspikeMultipleItems: any; - unspikeItem: (item, stage?: string) => wdpromise.Promise; + unspikeItem: (item, stage?: string) => void; openItemMenu: (group: any, item: any) => ElementFinder; showMonitoringSettings: () => void; setLabel: (label: any) => void; @@ -98,7 +97,6 @@ class Monitoring { openSendMenu: () => void; publish: () => void; getPublishButtonText: any; - startUpload: () => void; uploadModal: ElementFinder; openFetchAsOptions: (group: any, item: any) => void; clickOnFetchButton: any; @@ -123,7 +121,6 @@ class Monitoring { searchInput: ElementFinder; getCorrectionItems: (group: any) => any; getTakeItems: (group: any) => any; - getSendToDropdown: () => ElementFinder; getPackageItem: (index: any) => ElementFinder; getPackageItemActionDropdown: (index: any) => ElementFinder; getPackageItemLabelEntry: () => ElementFinder; @@ -162,22 +159,13 @@ class Monitoring { el(['workspace-navigation'], by.css('[aria-label="Search"]')).click(); }; - /** - * On monitoring view create a new item - * - * @param {string} action - the create item action can be: create_text_article, - * create_preformatted_article and create_package - */ - this.createItemAction = function(action) { - element(by.className('icon-plus-large')).click(); - element(by.id(action)).click(); - browser.sleep(500); - }; - /** * Create new item using desk template */ - this.createFromDeskTemplate = () => this.createItemAction('create_text_article'); + this.createFromDeskTemplate = () => { + el(['content-create']).click(); + el(['content-create-dropdown', 'default-desk-template']).click(); + }; this.getGroup = function(group: number) { return this.getGroups().get(group); @@ -187,7 +175,7 @@ class Monitoring { const groups = element.all(by.repeater('group in aggregate.groups')); browser.sleep(3000); // due to debouncing, loading does not start immediately - browser.wait(ECE.hasElementCount(els(['item-list--loading']), 0), 2000); + browser.wait(ECE.hasElementCount(els(['item-list--loading']), 0), 3000); return groups; }; @@ -512,19 +500,20 @@ class Monitoring { this.unspikeMultipleItems = function() { multiAction('Unspike'); - return element(by.buttonText('send')).click(); + el(['interactive-actions-panel', 'unspike']).click(); }; this.unspikeItem = function(item, stage?: string) { articleList.executeContextMenuAction(this.getSpikedItem(item), 'Unspike Item'); - var sidebar = element.all(by.css('.side-panel')).last(); - if (stage) { - sidebar.element(by.buttonText(stage)).click(); + el( + ['interactive-actions-panel', 'stage-select'], + by.cssContainingText('[data-test-id="item"]', stage), + ).click(); } - return element(by.buttonText('send')).click(); + el(['interactive-actions-panel', 'unspike']).click(); }; this.openItemMenu = function(group, item) { @@ -539,7 +528,7 @@ class Monitoring { }; this.showMonitoringSettings = function() { - element(by.css('.icon-settings')).click(); + el(['monitoring-settings-button']).click(); browser.wait(() => element.all(by.css('.aggregate-widget-config')).isDisplayed()); element.all(by.css('[ng-click="goTo(step)"]')).first().click(); }; @@ -717,7 +706,7 @@ class Monitoring { }; this.openCreateMenu = function() { - element(by.className('sd-create-btn')).click(); + element(by.css('[data-test-id="content-create"]')).click(); browser.sleep(100); }; @@ -733,10 +722,6 @@ class Monitoring { this.getPublishButtonText = () => element(by.css('[ng-click="publish()"]')).getText(); - this.startUpload = function() { - element(by.id('start-upload-btn')).click(); - }; - this.uploadModal = element(by.className('upload-media')); this.openFetchAsOptions = function(group, item) { @@ -744,7 +729,7 @@ class Monitoring { }; this.clickOnFetchButton = function() { - return element(by.css('[ng-click="send()"]')).click(); + el(['interactive-actions-panel', 'fetch']).click(); }; // Cancel button resets the multi selection @@ -763,7 +748,7 @@ class Monitoring { this.fetchAndOpen = function(group, item) { this.actionOnItem('Fetch To', group, item); - return element(by.css('[ng-click="send(true)"]')).click(); + el(['interactive-actions-panel', 'fetch-and-open']).click(); }; /** @@ -976,18 +961,6 @@ class Monitoring { return this.getGroupItems(group).all(by.className('takekey')); }; - /** - * Returns the desk dropdown in send to panel - * - */ - this.getSendToDropdown = () => { - var sidebar = element.all(by.css('.side-panel')).last(), - dropdown = sidebar.element(by.css('.dropdown--boxed .dropdown__toggle')), - dropdownSelected = dropdown.element(by.css('[ng-show="selectedDesk"]')); - - return dropdownSelected; - }; - this.getPackageItem = function(index) { var elemIndex = index ? index : 0; diff --git a/e2e/client/specs/helpers/search.ts b/e2e/client/specs/helpers/search.ts index 08b972c092..ae0b6bfe16 100644 --- a/e2e/client/specs/helpers/search.ts +++ b/e2e/client/specs/helpers/search.ts @@ -149,7 +149,8 @@ class GlobalSearch { itemElem.click(); - browser.wait(() => itemElem.getAttribute('class').then((classes) => classes.includes('active')), 500); + browser.wait(ECE.attributeContains(itemElem, 'class', 'active')); + browser.sleep(350); // there is timeout on click }; @@ -181,7 +182,7 @@ class GlobalSearch { browser.sleep(100); const btn = itemElem.element(by.className('icn-btn')); - browser.wait(ECE.elementToBeClickable(btn), 1000); + browser.wait(ECE.elementToBeClickable(btn)); btn.click(); const menu = element(by.css('.dropdown__menu.open')); diff --git a/e2e/client/specs/helpers/templates.ts b/e2e/client/specs/helpers/templates.ts index 2cb5222e19..5e09f0d313 100644 --- a/e2e/client/specs/helpers/templates.ts +++ b/e2e/client/specs/helpers/templates.ts @@ -38,6 +38,7 @@ class Templates { getRow: (name: any) => any; getValidationElement: any; getListCount: () => any; + selectProfile: (name: string) => void; constructor() { /** List of templates on template settings list **/ @@ -300,6 +301,10 @@ class Templates { this.getListCount = function() { return this.list.count(); }; + + this.selectProfile = (name) => { + element(by.id('template-profile')).element(by.cssContainingText('option', name)).click(); + }; } } diff --git a/e2e/client/specs/helpers/tree-select-driver.ts b/e2e/client/specs/helpers/tree-select-driver.ts new file mode 100644 index 0000000000..d5fb74b211 --- /dev/null +++ b/e2e/client/specs/helpers/tree-select-driver.ts @@ -0,0 +1,51 @@ +import {by, element, ElementFinder, promise, protractor} from 'protractor'; +import {el, els, s} from '@superdesk/end-to-end-testing-helpers'; + +export class TreeSelectDriver { + private _element: ElementFinder; + + constructor(_element: ElementFinder) { + this._element = _element; + + this.getValue = this.getValue.bind(this); + this.addValue = this.addValue.bind(this); + this.setValue = this.setValue.bind(this); + } + + getValue(): promise.Promise> { + return els(['item'], null, this._element).then((elements) => { + return protractor.promise.all( + elements.map((_el: ElementFinder) => { + return _el.getAttribute('innerText'); + }), + ); + }); + } + + addValue(value: string | Array): void { + el(['open-popover'], null, this._element).click(); + + const values = typeof value === 'string' ? [value] : value; + + for (let i = 0; i < values.length; i++) { + element( + by.cssContainingText( + '[data-test-id="tree-select-popover"] [data-test-id="option"]', + values[i], + ), + ).click(); + } + } + + setValue(value: string | Array) { + const maybeClearButton = this._element.element(s(['clear-value'])); + + maybeClearButton.isPresent().then((present) => { + if (present === true) { + maybeClearButton.click(); + } + }); + + this.addValue(value); + } +} diff --git a/e2e/client/specs/helpers/utils.ts b/e2e/client/specs/helpers/utils.ts index e24adcebe7..f3725748c0 100644 --- a/e2e/client/specs/helpers/utils.ts +++ b/e2e/client/specs/helpers/utils.ts @@ -179,13 +179,6 @@ export function altKey(key) { export function assertToastMsg(type: 'info' | 'success' | 'error', msg: string) { const elem = element(s([`notification--${type}`], msg)); - click(elem); - - /** - * It seems there's an issue with protractor: - * Clicking an element throws `StaleElementReferenceError` when clicked immediately - * after waiting until it's clickable. - */ elem.isPresent().then((present) => { // Only click if the toast is still present. if (present) { diff --git a/e2e/client/specs/internal_destinations_spec.ts b/e2e/client/specs/internal_destinations_spec.ts index 635932d453..8d81f53016 100644 --- a/e2e/client/specs/internal_destinations_spec.ts +++ b/e2e/client/specs/internal_destinations_spec.ts @@ -1,6 +1,6 @@ /* eslint-disable newline-per-chained-call */ -import {browser, element, by} from 'protractor'; +import {browser, element, by, ExpectedConditions as EC} from 'protractor'; import {el, els, s, ECE, hover} from '@superdesk/end-to-end-testing-helpers'; import {nav} from './helpers/utils'; @@ -166,16 +166,17 @@ describe('internal destinations & generic-page-list', () => { el(['list-page--filters-form', 'gform-input--desk']).click(); el(['list-page--filters-form', 'gform-input--desk'], by.buttonText('Sports Desk')).click(); el(['list-page--filters-form', 'filters-submit']).click(); - browser.wait(ECE.hasElementCount(items, 2)); + + browser.wait(ECE.hasElementCount(items, 2), 1000); expect(els(['list-page--filters-active', 'tag-label']).count()).toBe(1); var activeFilter = els(['list-page--filters-active', 'tag-label']).get(0); - browser.wait(ECE.attributeEquals(activeFilter, 'textContent', 'desk: Sports Desk')); + browser.wait(EC.textToBePresentInElement(activeFilter, 'Desk: Sports Desk'), 1000); el(['tag-label--remove'], null, activeFilter).click(); - browser.wait(ECE.hasElementCount(items, 3)); + browser.wait(ECE.hasElementCount(items, 3), 1000); expect(els(['list-page--filters-active', 'tag-label']).count()).toBe(0); }); }); diff --git a/e2e/client/specs/legal_archive_spec.ts b/e2e/client/specs/legal_archive_spec.ts index fd0ae6f285..5ccf09f610 100644 --- a/e2e/client/specs/legal_archive_spec.ts +++ b/e2e/client/specs/legal_archive_spec.ts @@ -79,7 +79,7 @@ describe('legal_archive', () => { expect(authoring.edit_button.isPresent()).toBe(false); expect(authoring.edit_correct_button.isPresent()).toBe(false); expect(authoring.edit_kill_button.isPresent()).toBe(false); - expect(authoring.navbarMenuBtn.isPresent()).toBe(false); + expect(element(by.css('[data-test-id="content-create"]')).isPresent()).toBe(false); expect(authoring.sendToButton.isDisplayed()).toBe(false); authoring.showInfo(); diff --git a/e2e/client/specs/monitoring_spec.ts b/e2e/client/specs/monitoring_spec.ts index 90f77a4ede..ece9da142e 100644 --- a/e2e/client/specs/monitoring_spec.ts +++ b/e2e/client/specs/monitoring_spec.ts @@ -1,19 +1,19 @@ /* eslint-disable newline-per-chained-call */ -import {element, browser, by, protractor, WebElementPromise, ElementFinder} from 'protractor'; +import {element, browser, by, protractor, ElementFinder} from 'protractor'; import {monitoring} from './helpers/monitoring'; import {workspace} from './helpers/workspace'; import {authoring} from './helpers/authoring'; import {dashboard} from './helpers/dashboard'; +import {TreeSelectDriver} from './helpers/tree-select-driver'; import {desks} from './helpers/desks'; import {el, s, els, ECE, articleList, getFocusedElement} from '@superdesk/end-to-end-testing-helpers'; import {nav} from './helpers/utils'; function createItem(headline: string) { - el(['content-create']).click(); - el(['content-create-dropdown']).element(by.buttonText('More templates...')).click(); - el(['select-template'], by.buttonText('editor3 template')).click(); + authoring.createTextItemFromTemplate('editor3 template'); + browser.wait(ECE.visibilityOf(element(s(['authoring'])))); el(['authoring', 'field--headline'], by.css('[contenteditable]')).sendKeys(headline); @@ -249,7 +249,6 @@ describe('monitoring', () => { it('configure a saved search from other user', () => { monitoring.openMonitoring(); workspace.createWorkspace('My Workspace'); - browser.sleep(500); monitoring.showMonitoringSettings(); monitoring.nextStages(); monitoring.toggleGlobalSearch(3); @@ -406,7 +405,7 @@ describe('monitoring', () => { authoring.close(); browser.wait(ECE.hasElementCount(els(['article-item']), 3), 2000); el(['content-profile-dropdown']).click(); - browser.wait(ECE.hasElementCount(els(['content-profiles']), 2)); + browser.wait(ECE.hasElementCount(els(['content-profiles']), 3)); el(['content-profile-dropdown'], by.buttonText('testing')).click(); browser.wait(ECE.hasElementCount(els(['article-item']), 1)); expect(monitoring.getTextItemBySlugline(0, 0)).toBe('TESTING1 SLUGLINE'); @@ -421,7 +420,7 @@ describe('monitoring', () => { expect(monitoring.getTextItem(4, 0)).toBe('item4'); el(['content-profile-dropdown']).click(); - browser.wait(ECE.hasElementCount(els(['content-profiles']), 2)); + browser.wait(ECE.hasElementCount(els(['content-profiles']), 3)); el(['content-profile-dropdown'], by.buttonText('testing')).click(); browser.wait(ECE.hasElementCount(els(['article-item']), 1), 2000); expect(monitoring.getTextItemBySlugline(0, 0)).toBe('TESTING1 SLUGLINE'); @@ -473,8 +472,10 @@ describe('monitoring', () => { it('can start content upload', () => { monitoring.openMonitoring(); - monitoring.openCreateMenu(); - monitoring.startUpload(); + + el(['content-create']).click(); + el(['content-create-dropdown', 'upload-media']).click(); + expect(monitoring.uploadModal.isDisplayed()).toBeTruthy(); }); @@ -667,7 +668,7 @@ describe('monitoring', () => { monitoring.fetchAndOpen(0, 5); - expect(authoring.save_button.isDisplayed()).toBe(true); + browser.wait(ECE.visibilityOf(authoring.save_button)); }); it('can display desk content in desk single view with their respective titles', () => { @@ -773,7 +774,6 @@ describe('monitoring', () => { monitoring.saveSettings(); monitoring.openMonitoring(); - browser.sleep(3000); // wait for monitoring groups to load expect(monitoring.getTextItem(1, 2)).toBe('item6'); @@ -791,7 +791,8 @@ describe('monitoring', () => { monitoring.actionOnItemSubmenu('Publishing actions', 'Correct item', 0, 0); authoring.send_correction_button.click(); - expect(element(by.id('multi-select-count')).isPresent()).toBeFalsy(); + + browser.wait(ECE.stalenessOf(element(by.id('multi-select-count')))); }); it('can view published duplicated item in duplicate tab of non-published original item', () => { @@ -839,18 +840,23 @@ describe('monitoring', () => { authoring.duplicateTo('Sports Desk', 'one'); monitoring.actionOnItemSubmenu('Duplicate', 'Duplicate To', 2, 0); - var dropdownSelected = monitoring.getSendToDropdown(); + expect( + new TreeSelectDriver( + el(['interactive-actions-panel', 'destination-select']), + ).getValue(), + ).toEqual(['Sports Desk']); - browser.sleep(500); - expect(dropdownSelected.getText()).toEqual('Sports Desk'); authoring.duplicateTo('Politic Desk', 'two', true); monitoring.actionOnItemSubmenu('Duplicate', 'Duplicate To', 2, 0); - - dropdownSelected = monitoring.getSendToDropdown(); authoring.close(); browser.sleep(500); - expect(dropdownSelected.getText()).toEqual('Politic Desk'); + + expect( + new TreeSelectDriver( + el(['interactive-actions-panel', 'destination-select']), + ).getValue(), + ).toEqual(['Politic Desk']); }); it('can view published item as readonly when opened', () => { @@ -866,11 +872,10 @@ describe('monitoring', () => { monitoring.actionOnItem('Open', 4, 0); expect(authoring.save_button.isPresent()).toBe(false); // Save button hidden for publish item - var textField = element(by.className('text-editor')); - // expect contenteditable=true attribute is missing/null for text-editor field, - // hence editing is disabled for published item + const bodyHtml = element(by.css('.field.body [contenteditable]')); - expect(textField.getAttribute('contenteditable')).toBe(null); + expect(bodyHtml.isPresent()).toBe(true); + expect(bodyHtml.getAttribute('contenteditable')).toBe('false'); }); it('closes preview when an item is opened for editing', () => { @@ -898,11 +903,12 @@ describe('monitoring', () => { it('Can create items from templates', () => { const slugline = 'slugline template'; const editorsNote = 'test editor\'s note for template'; + const newTemplateName = 'template 1234'; monitoring.openMonitoring(); expect(browser.isElementPresent(element(s(['authoring'])))).toBe(false); - el(['content-create']).click(); - el(['content-create-dropdown']).element(by.buttonText('Plain text')).click(); + + authoring.createTextItemFromTemplate('plain text'); expect(browser.isElementPresent(element(s(['authoring'])))).toBe(true); el(['authoring', 'field-slugline']).sendKeys(slugline); @@ -912,13 +918,16 @@ describe('monitoring', () => { el(['authoring', 'actions-button']).click(); el(['authoring', 'actions-list']).element(by.buttonText('Save as template')).click(); + + el(['save-as-template', 'name-input']).clear(); + el(['save-as-template', 'name-input']).sendKeys(newTemplateName); + el(['create-template-modal--save']).click(); + el(['authoring', 'close']).click(); expect(browser.isElementPresent(element(s(['authoring'])))).toBe(false); - el(['content-create']).click(); - el(['content-create-dropdown']).element(by.buttonText('More templates...')).click(); - el(['select-template'], by.buttonText(slugline)).click(); + authoring.createTextItemFromTemplate(newTemplateName); browser.sleep(500); // animation expect(browser.isElementPresent(element(s(['authoring'])))).toBe(true); diff --git a/e2e/client/specs/publishing_spec.ts b/e2e/client/specs/publishing_spec.ts index 4d9802b3b0..335e159573 100644 --- a/e2e/client/specs/publishing_spec.ts +++ b/e2e/client/specs/publishing_spec.ts @@ -9,6 +9,7 @@ import {workspace} from './helpers/workspace'; import {authoring} from './helpers/authoring'; import {el, els, ECE} from '@superdesk/end-to-end-testing-helpers'; import {executeContextMenuAction} from '@superdesk/end-to-end-testing-helpers/dist/articlesList'; +import {TreeSelectDriver} from './helpers/tree-select-driver'; describe('publishing', () => { beforeEach(monitoring.openMonitoring); @@ -61,8 +62,9 @@ describe('publishing', () => { executeContextMenuAction(els(['article-item'], null, thirdStage).get(0), 'Edit'); el(['authoring', 'open-send-publish-pane']).click(); - el(['authoring', 'send-publish-pane', 'tab--publish']).click(); - el(['authoring', 'send-publish-pane', 'publish']).click(); + + el(['authoring', 'interactive-actions-panel', 'tabs'], by.buttonText('Publish')).click(); + el(['authoring', 'interactive-actions-panel', 'publish']).click(); assertToastMsg('error', 'SUBJECT is a required field'); assertToastMsg('error', 'BODY HTML is a required field'); @@ -81,9 +83,7 @@ describe('publishing', () => { it('can send and publish', () => { workspace.selectDesk('Politic Desk'); - el(['content-create']).click(); - el(['content-create-dropdown']).element(by.buttonText('More templates...')).click(); - el(['select-template'], by.buttonText('testing')).click(); + authoring.createTextItemFromTemplate('testing'); const slugline = 'testing-send-and-publish'; @@ -91,19 +91,20 @@ describe('publishing', () => { el(['authoring', 'save']).click(); el(['authoring', 'open-send-publish-pane']).click(); - el(['authoring', 'send-publish-pane', 'tab--publish']).click(); - el(['authoring', 'send-publish-pane', 'publish-from--options', 'desk-select--handle']).click(); - el( - ['authoring', 'send-publish-pane', 'publish-from--options', 'desk-select--options'], - by.buttonText('Sports Desk'), - ).click(); + el(['authoring', 'interactive-actions-panel', 'tabs'], by.buttonText('Publish')).click(); + + new TreeSelectDriver( + el(['interactive-actions-panel', 'destination-select']), + ).setValue('Sports Desk'); - el(['authoring', 'send-publish-pane', 'publish-from--submit']).click(); + el(['authoring', 'interactive-actions-panel', 'publish-from']).click(); assertToastMsg('success', 'Item published.'); - browser.wait(ECE.stalenessOf(element(by.cssContainingText( + const firstGroup = els(['monitoring-group']).get(0); + + browser.wait(ECE.stalenessOf(firstGroup.element(by.cssContainingText( '[data-test-id="article-item"] [data-test-id="field--slugline"]', slugline, ))), MONITORING_DEBOUNCE_MAX_WAIT); diff --git a/e2e/client/specs/search_spec.ts b/e2e/client/specs/search_spec.ts index 4875eda844..e89a122047 100644 --- a/e2e/client/specs/search_spec.ts +++ b/e2e/client/specs/search_spec.ts @@ -7,7 +7,7 @@ import {globalSearch} from './helpers/search'; import {content} from './helpers/content'; import {authoring} from './helpers/authoring'; import {nav, scrollToView} from './helpers/utils'; -import {ECE} from '@superdesk/end-to-end-testing-helpers'; +import {ECE, el} from '@superdesk/end-to-end-testing-helpers'; describe('search', () => { beforeEach(() => { @@ -126,7 +126,7 @@ describe('search', () => { authoring.sendTo('Politic Desk'); authoring.confirmSendTo(); monitoring.switchToDesk('POLITIC DESK'); - expect(monitoring.getTextItem(1, 0)).toBe('From-Sports-To-Politics'); + expect(monitoring.getTextItem(0, 0)).toBe('From-Sports-To-Politics'); // search by from desk field globalSearch.openGlobalSearch(); @@ -228,21 +228,9 @@ describe('search', () => { browser.actions().sendKeys(protractor.Key.UP).perform(); expect(previewPane.isPresent()).toBe(false); // UP arrow key avoided for opening preview // it should not effect global keyboard shortcuts (e.g: 'ctrl+alt+d', 'ctrl+shift+*') - // now test 'ctrl+shift+*' shortcut that triggers spell checker when not set to automatic - expect(element(by.model('spellcheckMenu.isAuto')).getAttribute('checked')).toBeTruthy(); - authoring.toggleAutoSpellCheck(); - expect(element(by.model('spellcheckMenu.isAuto')).getAttribute('checked')).toBeFalsy(); - authoring.focusBodyHtmlElement(); - browser.actions().sendKeys(protractor.Key.ENTER).perform(); - browser.actions().sendKeys('Testhilite').perform(); - expect(authoring.getBodyText()).toContain('Testhilite'); - expect(authoring.getBodyInnerHtml()).not.toContain('sderror sdhilite'); - // trigger spell checker via keyboard operation - browser.actions().sendKeys(protractor.Key.chord(protractor.Key.CONTROL, protractor.Key.SHIFT, 'y')).perform(); - expect(authoring.getBodyText()).toContain('Testhilite'); - expect(authoring.getBodyInnerHtml()).toContain('sderror sdhilite'); authoring.save(); authoring.close(); + // now test 'ctrl+0' shortcut that triggers story search dialog box browser.actions().sendKeys(protractor.Key.chord(protractor.Key.CONTROL, '0')).perform(); browser.sleep(200); @@ -250,10 +238,11 @@ describe('search', () => { expect(storyNameEl.isPresent()).toBe(true); storyNameEl.click(); - browser.actions().sendKeys('item1-in-archived').perform(); + browser.actions().sendKeys('item4').perform(); browser.actions().sendKeys(protractor.Key.ENTER).perform(); browser.sleep(200); - expect(authoring.getHeaderSluglineText()).toBe('item1 slugline'); + + expect(authoring.getHeaderSluglineText()).toBe('item4 slugline'); authoring.close(); }); @@ -285,14 +274,21 @@ describe('search', () => { authoring.closeSendAndPublish(); authoring.save(); authoring.close(); - expect(globalSearch.getItem(0).element(by.className('state_embargo')).isDisplayed()).toBe(true); - expect(globalSearch.getItem(0).element(by.className('state_embargo')).getText()).toEqual('EMBARGO'); + + const embargoElement = globalSearch.getItem(0).element(by.className('state_embargo')); + + browser.wait(ECE.visibilityOf(embargoElement)); + + expect(embargoElement.getText()).toEqual('EMBARGO'); }); it('can search scheduled', () => { globalSearch.waitForItemCount(16); globalSearch.actionOnItem('Edit', 'item9'); authoring.schedule(false); + + browser.wait(ECE.stalenessOf(el(['notification--success']))); + globalSearch.openFilterPanel(); globalSearch.openParameters(); globalSearch.toggleSearchTabs('filters'); diff --git a/e2e/client/specs/send_spec.ts b/e2e/client/specs/send_spec.ts index cd94f35b93..4bd82767e4 100644 --- a/e2e/client/specs/send_spec.ts +++ b/e2e/client/specs/send_spec.ts @@ -6,6 +6,8 @@ import {monitoring} from './helpers/monitoring'; import {workspace} from './helpers/workspace'; import {content} from './helpers/content'; import {authoring} from './helpers/authoring'; +import {ECE, el} from '@superdesk/end-to-end-testing-helpers'; +import {TreeSelectDriver} from './helpers/tree-select-driver'; describe('send', () => { function getItemState(index) { @@ -32,14 +34,28 @@ describe('send', () => { expect(getItemState(0)).toBe('SUBMITTED'); }); - it('warns that there are spelling mistakes', () => { + /** + * Not sure if this was ever testing the intended thing. + * Currently it is only passing by accident + * since there's a modal that prompts to save unsaved changes. + * It isn't testing anything related to spellchecking. + * I'm disabling it in case we wanted to reimplement it in the future. + */ + xit('warns that there are spelling mistakes', () => { workspace.editItem(1); authoring.writeText('mispeled word'); authoring.sendTo('Sports Desk'); expect(element(by.className('modal__content')).isDisplayed()).toBe(true); }); - it('can submit item to a desk although there are spelling mistakes', () => { + /** + * Not sure if this was ever testing the intended thing. + * Currently it is only passing by accident + * since there's a modal that prompts to save unsaved changes. + * It isn't testing anything related to spellchecking. + * I'm disabling it in case we wanted to reimplement it in the future. + */ + xit('can submit item to a desk although there are spelling mistakes', () => { workspace.editItem(1); authoring.writeText('mispeled word'); authoring.sendTo('Sports Desk'); @@ -55,7 +71,14 @@ describe('send', () => { expect(getItemState(0)).toBe('SUBMITTED'); }); - it('can cancel submit request because there are spelling mistakes', () => { + /** + * Not sure if this was ever testing the intended thing. + * Currently it is only passing by accident + * since there's a modal that prompts to save unsaved changes. + * It isn't testing anything related to spellchecking. + * I'm disabling it in case we wanted to reimplement it in the future. + */ + xit('can cancel submit request because there are spelling mistakes', () => { workspace.editItem(1); authoring.writeText('mispeled word'); authoring.sendTo('Sports Desk'); @@ -72,7 +95,8 @@ describe('send', () => { expect(monitoring.hasClass(element(by.id('main-container')), 'hideMonitoring')).toBe(true); authoring.sendToButton.click(); - expect(authoring.sendItemContainer.isDisplayed()).toBe(true); + + browser.wait(ECE.visibilityOf(el(['interactive-actions-panel']))); }); it('can display monitoring after submitting an item to a desk using full view of authoring', () => { @@ -126,12 +150,13 @@ describe('send', () => { monitoring.showHideList(); authoring.sendToButton.click(); - var sidebar = element.all(by.css('.side-panel')).last(), - dropdown = sidebar.element(by.css('.dropdown--boxed .dropdown__toggle')), - dropdownSelected = dropdown.element(by.css('[ng-show="selectedDesk"]')); + el(['authoring', 'interactive-actions-panel', 'tabs'], by.buttonText('Send to')).click(); - browser.sleep(500); - expect(dropdownSelected.getText()).toEqual('Politic Desk'); + expect( + new TreeSelectDriver( + el(['interactive-actions-panel', 'destination-select']), + ).getValue(), + ).toEqual(['Politic Desk']); }); it('can remember last sent destination desk and stage on multi selection sendTo panel', () => { @@ -154,15 +179,14 @@ describe('send', () => { // open sendTo panel monitoring.openSendMenu(); - var sidebar = element.all(by.css('.side-panel')).last(), - dropdown = sidebar.element(by.css('.dropdown--boxed .dropdown__toggle')), - dropdownSelected = dropdown.element(by.css('[ng-show="selectedDesk"]')); - - browser.sleep(100); - expect(dropdownSelected.getText()).toEqual('Sports Desk'); // desk remembered - - var btnStage = sidebar.element(by.buttonText('Working Stage')); + expect( + new TreeSelectDriver( + el(['interactive-actions-panel', 'destination-select']), + ).getValue(), + ).toEqual(['Sports Desk']); - expect(btnStage.getAttribute('class')).toContain('active'); // stage remembered + expect( + el(['interactive-actions-panel', 'stage-select']).getAttribute('data-test-value'), + ).toEqual('Working Stage'); }); }); diff --git a/e2e/client/specs/templates_spec.ts b/e2e/client/specs/templates_spec.ts index 9eeb2bcbab..6b14a20d21 100644 --- a/e2e/client/specs/templates_spec.ts +++ b/e2e/client/specs/templates_spec.ts @@ -21,6 +21,7 @@ describe('templates', () => { templates.openTemplatesSettings(); templates.add(); templates.getTemplateNameElement().sendKeys('New Template'); + templates.selectProfile('Plain text'); templates.setTemplateType('string:create'); templates.selectDesk('Politic Desk'); templates.selectDesk('Sports Desk'); @@ -43,14 +44,14 @@ describe('templates', () => { // check the New Template is accessable from both desks monitoring.openMonitoring(); workspace.selectDesk('Sports Desk'); - authoring.createTextItemFromTemplate('new'); + authoring.createTextItemFromTemplate('new template'); expect(authoring.getBodyText()).toBe('This is body from the template'); expect(authoring.getHeaderSluglineText()).toBe('Test Template'); expect(authoring.getHeadlineText()).toBe('New Item'); authoring.close(); workspace.selectDesk('Politic Desk'); - authoring.createTextItemFromTemplate('new'); + authoring.createTextItemFromTemplate('new template'); expect(authoring.getBodyText()).toBe('This is body from the template'); expect(authoring.getHeaderSluglineText()).toBe('Test Template'); expect(authoring.getHeadlineText()).toBe('New Item'); @@ -60,6 +61,7 @@ describe('templates', () => { templates.openTemplatesSettings(); templates.add(); templates.getTemplateNameElement().sendKeys('Second New Template'); + templates.selectProfile('Plain text'); templates.setTemplateType('string:create'); templates.selectDesk('Politic Desk'); templates.selectDesk('Sports Desk'); diff --git a/e2e/client/specs/users_spec.ts b/e2e/client/specs/users_spec.ts index dfc2d5cc80..b924feeabc 100644 --- a/e2e/client/specs/users_spec.ts +++ b/e2e/client/specs/users_spec.ts @@ -246,8 +246,8 @@ describe('users', () => { // navigate to Workspace and create a new article workspace.openContent(); - authoring.navbarMenuBtn.click(); - authoring.newPlainArticleLink.click(); + + authoring.createTextItem(); // authoring opened, click the set category menu and see what // categories are offered @@ -275,8 +275,7 @@ describe('users', () => { // navigate to Workspace and create a new article monitoring.openMonitoring(); - authoring.navbarMenuBtn.click(); - authoring.newPlainArticleLink.click(); + authoring.createTextItem(); browser.sleep(100); // Open subject metadata dropdown field @@ -354,19 +353,19 @@ describe('users', () => { }); it('while creating a new user', () => { - var buttonCreate = element(by.className('sd-create-btn')); + var buttonCreate = element(by.css('[data-test-id="create-user-button"]')); buttonCreate.click(); - expect(element(by.id('user_default_desk')).isPresent()).toBe(false); + expect(element(by.css('[data-test-id="default-desk-template"]')).isPresent()).toBe(false); }); it('while pre-viewing and user clicks on create new user', () => { - var buttonCreate = element(by.className('sd-create-btn')); + var buttonCreate = element(by.css('[data-test-id="create-user-button"]')); element.all(by.repeater('users')).first().click(); buttonCreate.click(); - expect(element(by.id('user_default_desk')).isPresent()).toBe(false); + expect(element(by.css('[data-test-id="default-desk-template"]')).isPresent()).toBe(false); }); }); diff --git a/e2e/server/requirements.in b/e2e/server/requirements.in index 682db2be94..9e53730db2 100644 --- a/e2e/server/requirements.in +++ b/e2e/server/requirements.in @@ -1,3 +1,3 @@ honcho gunicorn -git+https://github.com/superdesk/superdesk-core.git@develop#egg=Superdesk-Core +git+https://github.com/superdesk/superdesk-core.git@release/2.7#egg=Superdesk-Core diff --git a/e2e/server/requirements.txt b/e2e/server/requirements.txt index 5b416f0532..c30d71eaab 100644 --- a/e2e/server/requirements.txt +++ b/e2e/server/requirements.txt @@ -26,9 +26,9 @@ blinker==1.4 # flask-mail # raven # superdesk-core -boto3==1.28.37 +boto3==1.28.50 # via superdesk-core -botocore==1.31.37 +botocore==1.31.50 # via # boto3 # s3transfer @@ -71,7 +71,7 @@ click-repl==0.3.0 # via celery croniter==0.3.37 # via superdesk-core -cryptography==41.0.3 +cryptography==41.0.4 # via # authlib # jwcrypto @@ -167,7 +167,9 @@ oauthlib==3.2.2 packaging==23.1 # via gunicorn pillow==9.2.0 - # via superdesk-core + # via + # reportlab + # superdesk-core prompt-toolkit==3.0.39 # via click-repl pyasn1==0.5.0 @@ -180,6 +182,8 @@ pyasn1-modules==0.3.0 # via oauth2client pycparser==2.21 # via cffi +pyjwt==2.4.0 + # via superdesk-core pymongo==3.11.4 # via # eve @@ -199,7 +203,7 @@ python-magic==0.4.27 # via superdesk-core python-twitter==3.5 # via superdesk-core -pytz==2023.3 +pytz==2023.3.post1 # via # celery # eve-elastic @@ -216,6 +220,8 @@ redis==4.5.5 # superdesk-core regex==2020.7.14 # via superdesk-core +reportlab==3.6.13 + # via superdesk-core requests==2.31.0 # via # python-twitter @@ -237,9 +243,9 @@ six==1.16.0 # flask-oidc-ex # oauth2client # python-dateutil -superdesk-core @ git+https://github.com/superdesk/superdesk-core.git@develop +superdesk-core @ git+https://github.com/superdesk/superdesk-core.git@release/2.7 # via -r requirements.in -typing-extensions==4.7.1 +typing-extensions==4.8.0 # via superdesk-core tzlocal==2.1 # via superdesk-core diff --git a/end-to-end-testing-helpers/package-lock.json b/end-to-end-testing-helpers/package-lock.json index 4c6fc59cb8..b3cf7e7e16 100644 --- a/end-to-end-testing-helpers/package-lock.json +++ b/end-to-end-testing-helpers/package-lock.json @@ -2496,9 +2496,9 @@ "dev": true }, "typescript": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.0.3.tgz", - "integrity": "sha512-tEu6DGxGgRJPb/mVPIZ48e69xCn2yRmCgYmDugAVwmJ6o+0u1RI18eO7E7WBTLYLaEVVOhwQmcdhQHweux/WPg==", + "version": "4.5.2", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.5.2.tgz", + "integrity": "sha512-5BlMof9H1yGt0P8/WF+wPNw6GfctgGjXp5hkblpyT+8rkASSmkUKMXrxR0Xg8ThVCi/JnHQiKXeBaEwCeQwMFw==", "dev": true }, "unbox-primitive": { diff --git a/end-to-end-testing-helpers/package.json b/end-to-end-testing-helpers/package.json index 66228bb588..f5341a9f40 100644 --- a/end-to-end-testing-helpers/package.json +++ b/end-to-end-testing-helpers/package.json @@ -16,7 +16,7 @@ "devDependencies": { "protractor": "^7.0.0", "superdesk-code-style": "1.3.0", - "typescript": "4.0.3" + "typescript": "^4.5.2" }, "peerDependencies": { "protractor": "5.4.2" diff --git a/package-lock.json b/package-lock.json index 68e52ae7fb..11151ddba4 100644 --- a/package-lock.json +++ b/package-lock.json @@ -168,6 +168,11 @@ "resolved": "https://registry.npmjs.org/@popperjs/core/-/core-2.10.2.tgz", "integrity": "sha512-IXf3XA7+XyN7CP9gGh/XB0UxVMlvARGEgGXLubFICsUMGz6Q+DU+i4gGlpOxTjKvXjkJDJC8YdqdKkDj9qZHEQ==" }, + "@reach/observe-rect": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@reach/observe-rect/-/observe-rect-1.2.0.tgz", + "integrity": "sha512-Ba7HmkFgfQxZqqaeIWWkNK0rEhpxVQHIoVyW1YDSkGsGIXzcaW4deC8B0pZrNSSyLTdIk7y+5olKt5+g0GmFIQ==" + }, "@superdesk/build-tools": { "version": "file:build-tools", "dev": true, @@ -182,12 +187,85 @@ "rimraf": "3.0.2" }, "dependencies": { + "@types/glob": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==", + "dev": true, + "requires": { + "@types/minimatch": "*", + "@types/node": "*" + } + }, + "@types/minimatch": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.5.tgz", + "integrity": "sha512-Klz949h02Gz2uZCMGwDUSDS1YBlTdDDgbWHi+81l29tQALUtvz4rAYi5uoVhE5Lagoq6DeqAUlbrHvW/mXDgdQ==", + "dev": true + }, + "@types/node": { + "version": "17.0.7", + "resolved": "https://registry.npmjs.org/@types/node/-/node-17.0.7.tgz", + "integrity": "sha512-1QUk+WAUD4t8iR+Oj+UgI8oJa6yyxaB8a8pHaC8uqM6RrS1qbL7bf3Pwl5rHv0psm2CuDErgho6v5N+G+5fwtQ==", + "dev": true + }, + "@types/parse5": { + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/@types/parse5/-/parse5-5.0.3.tgz", + "integrity": "sha512-kUNnecmtkunAoQ3CnjmMkzNU/gtxG8guhi+Fk2U/kOpIKjIMKnXGp4IJCgQJrXSgMsWYimYG4TGjz/UzbGEBTw==", + "dev": true + }, + "ansi-styles": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-1.0.0.tgz", + "integrity": "sha1-yxAt8cVvUSPquLZ817mAJ6AnkXg=", + "dev": true + }, + "atob": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", + "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", + "dev": true + }, + "balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "chalk": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-0.4.0.tgz", + "integrity": "sha1-UZmj3c0MHv4jvAjBsCewYXbgxk8=", + "dev": true, + "requires": { + "ansi-styles": "~1.0.0", + "has-color": "~0.1.0", + "strip-ansi": "~0.1.0" + } + }, "commander": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", "dev": true }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", + "dev": true + }, "css": { "version": "2.2.4", "resolved": "https://registry.npmjs.org/css/-/css-2.2.4.tgz", @@ -200,6 +278,12 @@ "urix": "^0.1.0" } }, + "css-selector-parser": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/css-selector-parser/-/css-selector-parser-1.4.1.tgz", + "integrity": "sha512-HYPSb7y/Z7BNDCOrakL4raGO2zltZkbeXyAd6Tg9obzix6QhzxCotdBl6VT0Dv4vZfJGVz3WL/xaEI9Ly3ul0g==", + "dev": true + }, "css-selector-tokenizer": { "version": "0.7.1", "resolved": "https://registry.npmjs.org/css-selector-tokenizer/-/css-selector-tokenizer-0.7.1.tgz", @@ -232,6 +316,15 @@ "typescript": "2 - 4" } }, + "gettext-parser": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/gettext-parser/-/gettext-parser-1.1.0.tgz", + "integrity": "sha1-LFpmONiTk0ubVQN9CtgstwBLJnk=", + "dev": true, + "requires": { + "encoding": "^0.1.11" + } + }, "gettext.js": { "version": "0.9.0", "resolved": "https://registry.npmjs.org/gettext.js/-/gettext.js-0.9.0.tgz", @@ -241,12 +334,156 @@ "po2json": "^0.4.0" } }, + "glob": { + "version": "7.1.7", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz", + "integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "has-color": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/has-color/-/has-color-0.1.7.tgz", + "integrity": "sha1-ZxRKUmDDT8PMpnfQQdr1L+e3iy8=", + "dev": true + }, + "iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "requires": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + } + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "dev": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true + }, + "jsesc": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-0.5.0.tgz", + "integrity": "sha1-597mbjXW/Bb3EP6R1c9p9w8IkR0=", + "dev": true + }, "lodash": { "version": "4.17.19", "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.19.tgz", "integrity": "sha512-JNvd8XER9GQX0v2qJgsaN/mzFCNA5BRe/j8JN9d+tWyGLSodKQHKFicdwNYzWwI3wjRnaKPsGj1XkBjx/F96DQ==", "dev": true }, + "minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "nomnom": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/nomnom/-/nomnom-1.8.1.tgz", + "integrity": "sha1-IVH3Ikcrp55Qp2/BJbuMjy5Nwqc=", + "dev": true, + "requires": { + "chalk": "~0.4.0", + "underscore": "~1.6.0" + } + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "parse5": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-6.0.1.tgz", + "integrity": "sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw==", + "dev": true + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", + "dev": true + }, + "po2json": { + "version": "0.4.5", + "resolved": "https://registry.npmjs.org/po2json/-/po2json-0.4.5.tgz", + "integrity": "sha1-R7spUtoy1Yob4vJWpZjuvAt0URg=", + "dev": true, + "requires": { + "gettext-parser": "1.1.0", + "nomnom": "1.8.1" + } + }, + "pofile": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/pofile/-/pofile-1.0.11.tgz", + "integrity": "sha512-Vy9eH1dRD9wHjYt/QqXcTz+RnX/zg53xK+KljFSX30PvdDMb2z+c6uDUeblUGqqJgz3QFsdlA0IJvHziPmWtQg==", + "dev": true + }, + "regenerate": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.2.tgz", + "integrity": "sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A==", + "dev": true + }, + "regexpu-core": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-1.0.0.tgz", + "integrity": "sha1-hqdj9Y7k18L2sQLkdkBQ3n7ZDGs=", + "dev": true, + "requires": { + "regenerate": "^1.2.1", + "regjsgen": "^0.2.0", + "regjsparser": "^0.1.4" + } + }, + "regjsgen": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/regjsgen/-/regjsgen-0.2.0.tgz", + "integrity": "sha1-bAFq3qxVT3WCP+N6wFuS1aTtsfc=", + "dev": true + }, + "regjsparser": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.1.5.tgz", + "integrity": "sha1-fuj4Tcb6eS0/0K4ijSS9lJ6tIFw=", + "dev": true, + "requires": { + "jsesc": "~0.5.0" + } + }, + "resolve-url": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", + "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=", + "dev": true + }, "rimraf": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", @@ -256,11 +493,84 @@ "glob": "^7.1.3" } }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, "source-map": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", "dev": true + }, + "source-map-resolve": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.3.tgz", + "integrity": "sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==", + "dev": true, + "requires": { + "atob": "^2.1.2", + "decode-uri-component": "^0.2.0", + "resolve-url": "^0.2.1", + "source-map-url": "^0.4.0", + "urix": "^0.1.0" + } + }, + "source-map-url": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.1.tgz", + "integrity": "sha512-cPiFOTLUKvJFIg4SKVScy4ilPPW6rFgMgfuZJPNoDuMs3nC1HbMUycBoJw77xFIp6z1UJQJOfx6C9GMH80DiTw==", + "dev": true + }, + "strip-ansi": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-0.1.1.tgz", + "integrity": "sha1-OeipjQRNFQZgq+SmgIrPcLt7yZE=", + "dev": true + }, + "typescript": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.5.4.tgz", + "integrity": "sha512-VgYs2A2QIRuGphtzFV7aQJduJ2gyfTljngLzjpfW9FoYZF6xuw1W0vW9ghCKLfcWrCFxK81CSGRAvS1pn4fIUg==", + "dev": true + }, + "underscore": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/underscore/-/underscore-1.6.0.tgz", + "integrity": "sha1-izixDKze9jM3uLJOT/htRa6lKag=", + "dev": true + }, + "urix": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", + "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", + "dev": true + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", + "dev": true + } + } + }, + "@superdesk/common": { + "version": "0.0.17", + "resolved": "https://registry.npmjs.org/@superdesk/common/-/common-0.0.17.tgz", + "integrity": "sha512-DeycOche2WLJkH4k2JdG8IR5Oi3K3MRTrxXtDqlxLvp/lU9eqbp0glHuthnTnUPEfaUo6bsye6wmGKC6M0DGEQ==", + "requires": { + "date-fns": "2.7.0", + "lodash": "4.17.19", + "react": "16.9.0", + "react-dom": "16.9.0", + "react-sortable-hoc": "^1.11.0" + }, + "dependencies": { + "date-fns": { + "version": "2.7.0", + "resolved": "https://verdaccio.sourcefabric.org/date-fns/-/date-fns-2.7.0.tgz", + "integrity": "sha512-wxYp2PGoUDN5ZEACc61aOtYFvSsJUylIvCjpjDOqM1UDaKIIuMJ9fAnMYFHV3TQaDpfTVxhwNK/GiCaHKuemTA==" } } }, @@ -360,12 +670,19 @@ } }, "@types/draft-js": { - "version": "0.10.29", - "resolved": "https://registry.npmjs.org/@types/draft-js/-/draft-js-0.10.29.tgz", - "integrity": "sha512-SA9hhLBjYtBFYz5d7N8CMzDQ6Z4evRsTa6+oomJu2JMBtrGA3jIG2/RzaSfN3U/8tXCJd/vMntBvVMdlYupjpw==", + "version": "0.11.5", + "resolved": "https://registry.npmjs.org/@types/draft-js/-/draft-js-0.11.5.tgz", + "integrity": "sha512-rf33PMmHMOcTR4HyUcdheTB+1Wpmj5bMLIEDSHoXMAYnkYPO58TLcUxYdDy5chrGyzwNrsFHxwFX2NvbEOQVWA==", "requires": { "@types/react": "*", - "immutable": "^3.8.1" + "immutable": "~3.7.4" + }, + "dependencies": { + "immutable": { + "version": "3.7.6", + "resolved": "https://registry.npmjs.org/immutable/-/immutable-3.7.6.tgz", + "integrity": "sha512-AizQPcaofEtO11RZhPPHBOJRdo/20MKQF9mBLnVkBoyHi1/zXK8fzVdnEpSV9gxqtnh6Qomfp3F0xT5qP/vThw==" + } } }, "@types/enzyme": { @@ -385,16 +702,6 @@ "@types/enzyme": "*" } }, - "@types/glob": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.2.0.tgz", - "integrity": "sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==", - "dev": true, - "requires": { - "@types/minimatch": "*", - "@types/node": "*" - } - }, "@types/hoist-non-react-statics": { "version": "3.3.1", "resolved": "https://registry.npmjs.org/@types/hoist-non-react-statics/-/hoist-non-react-statics-3.3.1.tgz", @@ -409,23 +716,11 @@ "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.117.tgz", "integrity": "sha512-xyf2m6tRbz8qQKcxYZa7PA4SllYcay+eh25DN3jmNYY6gSTL7Htc/bttVdkqj2wfJGbeWlQiX8pIyJpKU+tubw==" }, - "@types/minimatch": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-5.1.2.tgz", - "integrity": "sha512-K0VQKziLUWkVKiRVrx4a40iPaxTUefQmjtkQofBkYRcoaaL/8rhwDWww9qWbrgicNOgnpIsMxyNIUM4+n6dUIA==", - "dev": true - }, "@types/node": { "version": "15.12.2", "resolved": "https://registry.npmjs.org/@types/node/-/node-15.12.2.tgz", "integrity": "sha512-zjQ69G564OCIWIOHSXyQEEDpdpGl+G348RAKY0XXy9Z5kU9Vzv1GMNnkar/ZJ8dzXB3COzD9Mo9NtRZ4xfgUww==" }, - "@types/parse5": { - "version": "5.0.3", - "resolved": "https://registry.npmjs.org/@types/parse5/-/parse5-5.0.3.tgz", - "integrity": "sha512-kUNnecmtkunAoQ3CnjmMkzNU/gtxG8guhi+Fk2U/kOpIKjIMKnXGp4IJCgQJrXSgMsWYimYG4TGjz/UzbGEBTw==", - "dev": true - }, "@types/prop-types": { "version": "15.7.3", "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.3.tgz", @@ -705,7 +1000,7 @@ "object-assign": { "version": "4.1.1", "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==" + "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" }, "prop-types": { "version": "15.8.1", @@ -1565,45 +1860,6 @@ "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.11.0.tgz", "integrity": "sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA==" }, - "axios": { - "version": "0.15.3", - "resolved": "https://registry.npmjs.org/axios/-/axios-0.15.3.tgz", - "integrity": "sha512-w3/VNaraEcDri16lbemQWQGKfaFk9O0IZkzKlLeF5r6WWDv9TkcXkP+MWkRK8FbxwfozY/liI+qtvhV295t3HQ==", - "dev": true, - "optional": true, - "requires": { - "follow-redirects": "1.0.0" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "optional": true, - "requires": { - "ms": "2.0.0" - } - }, - "follow-redirects": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.0.0.tgz", - "integrity": "sha512-7s+wBk4z5xTwVJuozRBAyRofWKjD3uG2CUjZfZTrw9f+f+z8ZSxOjAqfIDLtc0Hnz+wGK2Y8qd93nGGjXBYKsQ==", - "dev": true, - "optional": true, - "requires": { - "debug": "^2.2.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", - "dev": true, - "optional": true - } - } - }, "babel-code-frame": { "version": "6.26.0", "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", @@ -2376,7 +2632,7 @@ "strip-ansi": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-0.3.0.tgz", - "integrity": "sha512-DerhZL7j6i6/nEnVG0qViKXI0OKouvvpsAiaj7c+LfqZZZxdwZtv8+UiA/w4VUJpT8UzX0pR1dcHOii1GbmruQ==", + "integrity": "sha1-JfSOoiynkYfzF0pNuHWTR7sSYiA=", "requires": { "ansi-regex": "^0.2.1" } @@ -2384,7 +2640,7 @@ "supports-color": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-0.2.0.tgz", - "integrity": "sha512-tdCZ28MnM7k7cJDJc7Eq80A9CsRFAAOZUy41npOZCs++qSjfIy7o5Rh46CBk+Dk5FbKJ33X3Tqg4YrV07N5RaA==" + "integrity": "sha1-2S3iaU6z9nMjlz1649i1W0wiGQo=" } } }, @@ -3342,12 +3598,6 @@ "nth-check": "~1.0.1" } }, - "css-selector-parser": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/css-selector-parser/-/css-selector-parser-1.4.1.tgz", - "integrity": "sha512-HYPSb7y/Z7BNDCOrakL4raGO2zltZkbeXyAd6Tg9obzix6QhzxCotdBl6VT0Dv4vZfJGVz3WL/xaEI9Ly3ul0g==", - "dev": true - }, "css-selector-tokenizer": { "version": "0.7.1", "resolved": "https://registry.npmjs.org/css-selector-tokenizer/-/css-selector-tokenizer-0.7.1.tgz", @@ -3649,9 +3899,9 @@ "integrity": "sha512-u75aE0uAUogoo2u7PIN44vUfITrmtABDgy58XY1U9JZpr/2Ghhib8Y/w0Btot3tgSXNZL4iy9ROMsQ3P0dhYvw==" }, "defaults": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.4.tgz", - "integrity": "sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.3.tgz", + "integrity": "sha1-xlYFHpgX2f8I7YgUd/P+QBnz730=", "requires": { "clone": "^1.0.2" } @@ -3965,7 +4215,7 @@ "object-assign": { "version": "4.1.1", "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==" + "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" } } }, @@ -6182,7 +6432,7 @@ "object-assign": { "version": "4.1.1", "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==" + "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" } } }, @@ -8215,6 +8465,10 @@ } } }, + "htmldiff-js": { + "version": "github:dfoverdx/htmldiff-js#8cdfe9de9554536e0ac4238568af8db019066f75", + "from": "github:dfoverdx/htmldiff-js#master" + }, "htmlparser2": { "version": "3.10.1", "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-3.10.1.tgz", @@ -9709,6 +9963,16 @@ "streamroller": "0.7.0" }, "dependencies": { + "axios": { + "version": "0.15.3", + "resolved": "https://registry.npmjs.org/axios/-/axios-0.15.3.tgz", + "integrity": "sha512-w3/VNaraEcDri16lbemQWQGKfaFk9O0IZkzKlLeF5r6WWDv9TkcXkP+MWkRK8FbxwfozY/liI+qtvhV295t3HQ==", + "dev": true, + "optional": true, + "requires": { + "follow-redirects": "1.0.0" + } + }, "debug": { "version": "3.2.7", "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", @@ -9718,6 +9982,35 @@ "ms": "^2.1.1" } }, + "follow-redirects": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.0.0.tgz", + "integrity": "sha1-jjQpjL0uF28lTv/sdaHHjMhJ/Tc=", + "dev": true, + "optional": true, + "requires": { + "debug": "^2.2.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "optional": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true, + "optional": true + } + } + }, "ms": { "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", @@ -10873,16 +11166,6 @@ "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==" }, - "medium-editor": { - "version": "5.23.3", - "resolved": "https://registry.npmjs.org/medium-editor/-/medium-editor-5.23.3.tgz", - "integrity": "sha512-he9/TdjX8f8MGdXGfCs8AllrYnqXJJvjNkDKmPg3aPW/uoIrlRqtkFthrwvmd+u4QyzEiadhCCM0EwTiRdUCJw==" - }, - "medium-editor-tables": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/medium-editor-tables/-/medium-editor-tables-0.6.1.tgz", - "integrity": "sha512-JCbPpF2Sltb9nj6pldPpzmaWDI29dTfG49zu8c98jKSYEdaI+AFEZbflsynaJuQJvPn7GXkiYmy8O/iR8ad42A==" - }, "mem": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/mem/-/mem-1.1.0.tgz", @@ -11812,7 +12095,7 @@ "object-assign": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-3.0.0.tgz", - "integrity": "sha512-jHP15vXVGeVh1HuaA2wY6lxk+whK/x4KBG88VXeRma7CCun7iGD5qPc4eYykQ9sdQvg8jkwFKsSxHln2ybW3xQ==" + "integrity": "sha1-m+3VygiXlJvKR+f/QIBi1Un1h/I=" }, "object-component": { "version": "0.0.3", @@ -13337,7 +13620,7 @@ "object-assign": { "version": "4.1.1", "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==" } } }, @@ -13864,6 +14147,16 @@ "prop-types": "^15.5.10" } }, + "react-sortable-hoc": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/react-sortable-hoc/-/react-sortable-hoc-1.11.0.tgz", + "integrity": "sha512-v1CDCvdfoR3zLGNp6qsBa4J1BWMEVH25+UKxF/RvQRh+mrB+emqtVHMgZ+WreUiKJoEaiwYoScaueIKhMVBHUg==", + "requires": { + "@babel/runtime": "^7.2.0", + "invariant": "^2.2.4", + "prop-types": "^15.5.7" + } + }, "react-test-renderer": { "version": "16.14.0", "resolved": "https://registry.npmjs.org/react-test-renderer/-/react-test-renderer-16.14.0.tgz", @@ -13937,6 +14230,14 @@ } } }, + "react-virtual": { + "version": "2.10.4", + "resolved": "https://registry.npmjs.org/react-virtual/-/react-virtual-2.10.4.tgz", + "integrity": "sha512-Ir6+oPQZTVHfa6+JL9M7cvMILstFZH/H3jqeYeKI4MSUX+rIruVwFC6nGVXw9wqAw8L0Kg2KvfXxI85OvYQdpQ==", + "requires": { + "@reach/observe-rect": "^1.1.0" + } + }, "read-pkg": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-1.1.0.tgz", @@ -15831,7 +16132,7 @@ "strip-ansi": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", - "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", "requires": { "ansi-regex": "^2.0.0" } @@ -16373,9 +16674,9 @@ } }, "superdesk-ui-framework": { - "version": "3.0.54", - "resolved": "https://registry.npmjs.org/superdesk-ui-framework/-/superdesk-ui-framework-3.0.54.tgz", - "integrity": "sha512-wWtx2AEJUEShU7v60KteMcPW+vfP0iI3KDnWxnBxsNm6Y7T/nT2ROrd2U5dM/fjNt41jJEiP+AD7mN3ykX8Q4g==", + "version": "3.0.59", + "resolved": "https://registry.npmjs.org/superdesk-ui-framework/-/superdesk-ui-framework-3.0.59.tgz", + "integrity": "sha512-FuXyJNGVE970jlHWm0vD1Cr9QGLEfjONPaPfNSAKQZHW1f//By2VERgPi8TFv1kTdZOlXcOP6vFhnw/OR/Z6Nw==", "requires": { "@material-ui/lab": "^4.0.0-alpha.56", "@popperjs/core": "^2.4.0", @@ -16398,9 +16699,9 @@ }, "dependencies": { "@types/node": { - "version": "14.18.56", - "resolved": "https://registry.npmjs.org/@types/node/-/node-14.18.56.tgz", - "integrity": "sha512-+k+57NVS9opgrEn5l9c0gvD1r6C+PtyhVE4BTnMMRwiEA8ZO8uFcs6Yy2sXIy0eC95ZurBtRSvhZiHXBysbl6w==" + "version": "14.18.61", + "resolved": "https://registry.npmjs.org/@types/node/-/node-14.18.61.tgz", + "integrity": "sha512-1mFT4DqS4/s9tlZbdkwEB/EnSykA9MDeDLIk3FHApGvIMGY//qgstB2gu9GKGESWyW/qiRUO+jhlLJ9bBJ8j+Q==" }, "cheerio": { "version": "1.0.0-rc.12", @@ -16559,7 +16860,7 @@ "supports-color": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", - "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==" + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=" }, "svgo": { "version": "0.7.2", @@ -17437,7 +17738,7 @@ "core-util-is": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", - "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" + "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==" } } }, diff --git a/package.json b/package.json index 6065aae965..b17d5fac21 100644 --- a/package.json +++ b/package.json @@ -30,8 +30,9 @@ "dependencies": { "@metadata/exif": "github:superdesk/exif#431066d", "@popperjs/core": "2.10.2", + "@superdesk/common": "0.0.17", "@types/angular": "1.6.50", - "@types/draft-js": "0.10.29", + "@types/draft-js": "0.11.5", "@types/enzyme": "3.9.1", "@types/lodash": "4.14.117", "@types/react": "16.8.23", @@ -56,6 +57,7 @@ "classnames": "2.2.5", "css-loader": "0.28.10", "d3": "3.5.17", + "date-fns": "^2.7.0", "diff-match-patch": "1.0.0", "docs-soap": "github:tomaskikutis/docs-soap#convert-tables", "draft-js": "github:superdesk/draft-js#master", @@ -77,6 +79,7 @@ "grunt-webpack": "3.0.2", "hls.js": "0.12.4", "html-loader": "0.5.5", + "htmldiff-js": "github:dfoverdx/htmldiff-js#master", "immutable": "3.8.2", "jquery": "3.3.1", "jquery-jcrop": "0.9.13", @@ -88,8 +91,6 @@ "load-grunt-config": "0.19.2", "load-grunt-tasks": "3.5.2", "lodash": "4.17.19", - "medium-editor": "5.23.3", - "medium-editor-tables": "0.6.1", "ment.io": "0.9.23", "moment": "2.29.4", "moment-timezone": "0.5.41", @@ -112,17 +113,19 @@ "react-paginate": "6.3.0", "react-portal": "4.1.3", "react-redux": "7.2.1", + "react-sortable-hoc": "1.11.0", "react-textarea-autosize": "5.2.1", + "react-virtual": "2.10.4", "redux": "4.0.5", "redux-logger": "3.0.6", "redux-thunk": "2.3.0", "sass-loader": "6.0.6", "shortid": "2.2.8", "style-loader": "0.20.2", - "superdesk-ui-framework": "^3.0.54", + "superdesk-ui-framework": "^3.0.59", "ts-loader": "3.5.0", "tslint": "5.11.0", - "typescript": "~4.9.5", + "typescript": "4.9.5", "uuid": "8.3.1", "web-animations-js": "^2.3.2", "webpack": "3.11.0", diff --git a/patches/@types+draft-js++immutable+3.7.6.patch b/patches/@types+draft-js++immutable+3.7.6.patch new file mode 100644 index 0000000000..ce69329bcd --- /dev/null +++ b/patches/@types+draft-js++immutable+3.7.6.patch @@ -0,0 +1,49 @@ +diff --git a/node_modules/@types/draft-js/node_modules/immutable/dist/immutable.d.ts b/node_modules/@types/draft-js/node_modules/immutable/dist/immutable.d.ts +index f8eb1c6..87fe41e 100644 +--- a/node_modules/@types/draft-js/node_modules/immutable/dist/immutable.d.ts ++++ b/node_modules/@types/draft-js/node_modules/immutable/dist/immutable.d.ts +@@ -1982,7 +1982,7 @@ declare module Immutable { + * + */ + map( +- mapper: (value?: V, key?: K, iter?: /*this*/Iterable) => M, ++ mapper: (value: V, key: K, iter: /*this*/Iterable) => M, + context?: any + ): /*this*/Iterable; + +@@ -1995,7 +1995,7 @@ declare module Immutable { + * + */ + filter( +- predicate: (value?: V, key?: K, iter?: /*this*/Iterable) => boolean, ++ predicate: (value: V, key: K, iter: /*this*/Iterable) => boolean, + context?: any + ): /*this*/Iterable; + +@@ -2008,7 +2008,7 @@ declare module Immutable { + * + */ + filterNot( +- predicate: (value?: V, key?: K, iter?: /*this*/Iterable) => boolean, ++ predicate: (value: V, key: K, iter: /*this*/Iterable) => boolean, + context?: any + ): /*this*/Iterable; + +@@ -2044,7 +2044,7 @@ declare module Immutable { + * + */ + sortBy( +- comparatorValueMapper: (value?: V, key?: K, iter?: /*this*/Iterable) => C, ++ comparatorValueMapper: (value: V, key: K, iter: /*this*/Iterable) => C, + comparator?: (valueA: C, valueB: C) => number + ): /*this*/Iterable; + +@@ -2055,7 +2055,7 @@ declare module Immutable { + * Note: This is always an eager operation. + */ + groupBy( +- grouper: (value?: V, key?: K, iter?: /*this*/Iterable) => G, ++ grouper: (value: V, key: K, iter: /*this*/Iterable) => G, + context?: any + ): /*Map*/Seq.Keyed>; + diff --git a/scripts/api/article-duplicate.tsx b/scripts/api/article-duplicate.tsx new file mode 100644 index 0000000000..e5438e9f50 --- /dev/null +++ b/scripts/api/article-duplicate.tsx @@ -0,0 +1,50 @@ +import {IArticle} from 'superdesk-api'; +import {ISendToDestination} from 'core/interactive-article-actions-panel/interfaces'; +import {assertNever} from 'core/helpers/typescript-helpers'; +import {httpRequestJsonLocal} from 'core/helpers/network'; +import {notify} from 'core/notify/notify'; +import {gettextPlural} from 'core/utils'; +import {sdApi} from 'api'; +import ng from 'core/services/ng'; + +export function duplicateItems(items: Array, destination: ISendToDestination): Promise> { + return Promise.all( + items.map((item) => { + const payload = (() => { + if (destination.type === 'personal-space') { + return { + type: 'archive', + desk: null, + }; + } else if (destination.type === 'desk') { + return { + type: 'archive', + desk: destination.desk, + stage: destination.stage, + }; + } else { + assertNever(destination); + } + })(); + + return httpRequestJsonLocal({ + method: 'POST', + path: `/archive/${item._id}/duplicate`, + payload: payload, + }); + }), + ).then((res: Array) => { + notify.success(gettextPlural( + items.length, + 'Item duplicated', + 'Items duplicated', + )); + + sdApi.preferences.update('destination:active', destination); + + // TODO: Not sure if needed. Remove when monitoring view is moved to React. + ng.get('$rootScope').$broadcast('item:duplicate'); + + return res; + }); +} diff --git a/scripts/api/article-fetch.tsx b/scripts/api/article-fetch.tsx new file mode 100644 index 0000000000..ca47b06129 --- /dev/null +++ b/scripts/api/article-fetch.tsx @@ -0,0 +1,173 @@ +import React from 'react'; +import {IArticle} from 'superdesk-api'; +import {appConfig} from 'appConfig'; +import {ISendToDestinationDesk} from 'core/interactive-article-actions-panel/interfaces'; +import {httpRequestJsonLocal} from 'core/helpers/network'; +import {showModal} from '@superdesk/common'; +import {IModalSimpleAction, ModalSimple} from 'core/ui/components/modal-simple'; +import {gettext, gettextPlural} from 'core/utils'; +import {sdApi} from 'api'; + +function fetchFromIngest(item: IArticle, destination: ISendToDestinationDesk): Promise { + return httpRequestJsonLocal({ + method: 'POST', + path: `/ingest/${item._id}/fetch`, + payload: { + desk: destination.desk, + stage: destination.stage, + }, + }); +} + +function fetchFromExternalSource(item: IArticle, destination: ISendToDestinationDesk) { + return httpRequestJsonLocal({ + method: 'POST', + path: `/${item.fetch_endpoint}`, + payload: { + guid: item.guid, + desk: destination.desk, + stage: destination.stage, + }, + urlParams: { + repo: item.ingest_provider, + }, + }); +} + +interface IImageSizeValidationError { + name: string; + width: number; + height: number; +} + +function validateImageSize(items: Array) { + const validItems: Array = []; + const invalidItems: Array = []; + + items.forEach((item) => { + if (appConfig.pictures != null && item.type === 'picture' && item._type === 'ingest') { + const pictureWidth = item?.renditions.original.width; + const pictureHeight = item?.renditions.original.height; + + if (appConfig.pictures.minWidth > pictureWidth || appConfig.pictures.minHeight > pictureHeight) { + invalidItems.push({ + name: item.headline || item.slugline || 'image', + width: item.renditions.original.width, + height: item.renditions.original.height, + }); + } else { + validItems.push(item); + } + } else { + validItems.push(item); + } + }); + + return {validItems, invalidItems}; +} + +function doFetch(validatedItems: Array, selectedDestination: ISendToDestinationDesk) { + return Promise.all( + validatedItems.map((item) => { + if (item.fetch_endpoint != null) { + return fetchFromExternalSource(item, selectedDestination); + } else { + return fetchFromIngest(item, selectedDestination); + } + }), + ); +} + +export function fetchItems( + items: Array, + selectedDestination: ISendToDestinationDesk, +): Promise> { + return new Promise((resolve, reject) => { + const {validItems, invalidItems} = validateImageSize(items); + + if (invalidItems.length < 1) { + doFetch(validItems, selectedDestination).then(resolve); + } else { + showModal(({closeModal}) => { + let actions: Array = [ + { + label: gettext('Cancel'), + onClick: () => { + closeModal(); + reject(); + }, + }, + ]; + + if (validItems.length > 0) { + actions.push({ + label: gettext('Fetch valid({{count}})', {count: validItems.length}), + onClick: () => doFetch(validItems, selectedDestination).then(resolve), + }); + } + + return ( + +
+ { + gettextPlural( + invalidItems.length, + '{{n}} images can not be fetched because they are too small.', + 'One image can not be fetched because it is too small.', + {n: invalidItems.length}, + ) + ' ' + gettext( + 'Minimum allowed image size is {{minWidth}}x{{minHeight}}', + { + minWidth: appConfig.pictures.minWidth, + minHeight: appConfig.pictures.minHeight, + }, + ) + } + +

{gettext('Items that can not be fetched')}

+ + + + + + + + + + { + invalidItems.map(({name, width, height}, i) => ( + + + + + + )) + } + +
{gettext('file name')}{gettext('width')}{gettext('height')}
{name}{width}{height}
+
+
+ ); + }); + } + }); +} + +export function fetchItemsToCurrentDesk(items: Array) { + const currentDeskId = sdApi.desks.getCurrentDeskId(); + + return fetchItems( + items, + { + type: 'desk', + desk: currentDeskId, + stage: sdApi.desks.getDeskStages(currentDeskId).find( + (stage) => stage.default_incoming === true, + )._id, + }, + ); +} diff --git a/scripts/api/article-patch.ts b/scripts/api/article-patch.ts new file mode 100644 index 0000000000..d09960adb7 --- /dev/null +++ b/scripts/api/article-patch.ts @@ -0,0 +1,46 @@ +import {IArticle, IDangerousArticlePatchingOptions} from 'superdesk-api'; +import {sdApi} from 'api'; +import {extensions} from 'appConfig'; +import {dataApi} from 'core/helpers/CrudManager'; +import {dispatchInternalEvent} from 'core/internal-events'; +import {logger} from 'core/services/logger'; + +/** + * If article is locked in editing mode, it will apply the patch on top of existing changes + * and run the same code as if user edited a field manually. + * If article isn't locked, the patch will be sent directly to the server. + */ +export const patchArticle = ( + article: IArticle, + patch: Partial, + dangerousOptions?: IDangerousArticlePatchingOptions, +): Promise => { + const onPatchBeforeMiddlewares = Object.values(extensions) + .map((extension) => extension.activationResult?.contributions?.entities?.article?.onPatchBefore) + .filter((middleware) => middleware != null); + + return onPatchBeforeMiddlewares.reduce( + (current, next) => current.then((result) => next(article._id, result, dangerousOptions)), + Promise.resolve(patch), + ).then((patchFinal) => { + return dataApi.patchRaw( + // distinction between handling published and non-published items + // should be removed: SDESK-4687 + (sdApi.article.isPublished(article) ? 'published' : 'archive'), + article._id, + article._etag, + patchFinal, + ).then((res) => { + if (dangerousOptions?.patchDirectlyAndOverwriteAuthoringValues === true) { + dispatchInternalEvent( + 'dangerouslyOverwriteAuthoringData', + {...patch, _etag: res._etag, _id: res._id}, + ); + } + }); + }).catch((err) => { + if (err instanceof Error) { + logger.error(err); + } + }); +}; diff --git a/scripts/api/article-send.tsx b/scripts/api/article-send.tsx new file mode 100644 index 0000000000..965843e60b --- /dev/null +++ b/scripts/api/article-send.tsx @@ -0,0 +1,261 @@ +import React from 'react'; +import {IArticle, IDesk, ISuperdeskQuery, IRestApiResponse} from 'superdesk-api'; +import {ISendToDestination} from 'core/interactive-article-actions-panel/interfaces'; +import {sdApi} from 'api'; +import {extensions} from 'appConfig'; +import {notNullOrUndefined, assertNever} from 'core/helpers/typescript-helpers'; +import { + getPublishingDatePatch, + IPublishingDateOptions, +} from 'core/interactive-article-actions-panel/subcomponents/publishing-date-options'; +import {httpRequestJsonLocal} from 'core/helpers/network'; +import {notify} from 'core/notify/notify'; +import {gettext, getItemLabel} from 'core/utils'; +import {dispatchInternalEvent} from 'core/internal-events'; +import {toElasticQuery} from 'core/query-formatting'; +import {showModal} from '@superdesk/common'; +import {ModalSimple, IModalSimpleAction} from 'core/ui/components/modal-simple'; +import {UnorderedList} from 'core/ui/components/UnorderedList'; + +function getPublishedPackageItems(_package: IArticle): Promise> { + const query: ISuperdeskQuery = { + filter: {$and: [{'guid': {$in: sdApi.article.getPackageItemIds(_package)}}]}, + page: 0, + max_results: 200, + sort: [{'versioncreated': 'asc'}], + }; + + return httpRequestJsonLocal>({ + method: 'GET', + path: '/search', + urlParams: { + repo: 'published', + ...toElasticQuery(query), + }, + }).then((res) => res._items); +} + +function applyMiddlewares( + items: Array, + destination: ISendToDestination, +): Promise { + const selectedDeskObj: IDesk | null = (() => { + if (destination.type === 'desk') { + return sdApi.desks.getAllDesks().find((desk) => desk._id === destination.desk); + } else { + return null; + } + })(); + + const middlewares = Object.values(extensions) + .map((ext) => ext?.activationResult?.contributions?.entities?.article?.onSendBefore) + .filter(notNullOrUndefined); + + return middlewares.reduce( + (current, next) => { + return current.then(() => { + return next(items, selectedDeskObj); + }); + }, + Promise.resolve(), + ); +} + +function confirmSendingPackages(items: Array): Promise { + const packages = items.filter(({type}) => type === 'composite'); + + if (packages.length < 1) { + return Promise.resolve(); + } + + return new Promise((resolve, reject) => { + Promise.all( + packages.map((_package) => getPublishedPackageItems(_package).then((publishedPackageItems) => ({ + _package, + publishedPackageItems, + }))), + ).then((res) => { + const withPublishedItems = res.filter(({publishedPackageItems}) => publishedPackageItems.length > 0); + + if (withPublishedItems.length < 1) { + resolve(); + } else { + showModal(({closeModal}) => { + const actions: Array = [ + { + label: gettext('Cancel'), + onClick: () => { + closeModal(); + reject(); + }, + }, + { + label: gettext('Continue'), + onClick: () => { + closeModal(); + resolve(); + }, + primary: true, + }, + ]; + + return ( + + { + (() => { + if (withPublishedItems.length === 1) { + const _package = withPublishedItems[0]._package; + + return ( +
+

+ { + gettext( + 'The package "{{name}}" contains the following ' + + 'published items that can not be sent:', + { + name: getItemLabel(_package), + }, + ) + } +

+ + getItemLabel(item), + )} + /> +
+ ); + } else { + return ( +
+

+ { + gettext( + 'Some packages contain the following ' + + 'published items that can not be sent:', + ) + } +

+ +
+ + { + withPublishedItems.map(({_package, publishedPackageItems}) => ( +
+

{getItemLabel(_package)}

+ + getItemLabel(item), + )} + /> +
+ )) + } +
+ ); + } + })() + } +
+ ); + }); + } + }); + }); +} + +/** + * Promise may be rejected by middleware. + * Returns patches, not whole items. + */ +export function sendItems( + items: Array, + selectedDestination: ISendToDestination, + sendPackageItems: boolean = false, + publishingDateOptions?: IPublishingDateOptions, +): Promise>> { + return applyMiddlewares(items, selectedDestination) + .then(() => sendPackageItems ? confirmSendingPackages(items) : Promise.resolve()) + .then(() => { + return Promise.all( + items.map((item) => { + return (() => { + if (Object.keys(publishingDateOptions ?? {}).length < 1) { + return Promise.resolve({}); + } + + /** + * If needed, update embargo / publish schedule / time zone + */ + + var patch = getPublishingDatePatch(item, publishingDateOptions); + + if (Object.keys(patch).length > 0) { + return httpRequestJsonLocal({ + method: 'PATCH', + path: `/archive/${item._id}`, + payload: patch, + headers: { + 'If-Match': item._etag, + }, + }); + } else { + return Promise.resolve({}); + } + })().then((patch1: Partial) => { + const payload = (() => { + const basePayload = {}; + + if (sendPackageItems) { + basePayload['allPackageItems'] = true; + } + + if (selectedDestination.type === 'personal-space') { + return basePayload; + } else if (selectedDestination.type === 'desk') { + const _payload: Partial = { + ...basePayload, + task: { + desk: selectedDestination.desk, + stage: selectedDestination.stage, + }, + }; + + return _payload; + } else { + assertNever(selectedDestination); + } + })(); + + return httpRequestJsonLocal({ + method: 'POST', + path: `/archive/${item._id}/move`, + payload: payload, + }).then((patch2: Partial) => { + const patchFinal = { + ...patch1, + ...patch2, + }; + + // TODO: fix server response to contain correct links or none at all + delete patchFinal['_links']; + + return patchFinal; + }).catch((err) => { + notify.error(err._message ?? gettext('Unknown error occurred')); + + throw err; + }); + }); + }), + ); + }).then((patches: Array>) => { + sdApi.preferences.update('destination:active', selectedDestination); + notify.success(gettext('Sent successfully')); + + return patches; + }); +} diff --git a/scripts/api/article.ts b/scripts/api/article.ts index b2065a976c..1c07b756a8 100644 --- a/scripts/api/article.ts +++ b/scripts/api/article.ts @@ -1,34 +1,523 @@ -import {IArticle} from 'superdesk-api'; +import {sdApi} from 'api'; +import {appConfig, extensions} from 'appConfig'; +import {ITEM_STATE, KILLED_STATES, PUBLISHED_STATES} from 'apps/archive/constants'; +import {applicationState, openArticle} from 'core/get-superdesk-api-implementation'; +import {dataApi} from 'core/helpers/CrudManager'; +import {httpRequestJsonLocal} from 'core/helpers/network'; +import {assertNever} from 'core/helpers/typescript-helpers'; +import {copyJson} from 'core/helpers/utils'; +import { + IPublishingError, + ISendToDestination, + ISendToDestinationDesk, +} from 'core/interactive-article-actions-panel/interfaces'; +import {IPublishingDateOptions} from 'core/interactive-article-actions-panel/subcomponents/publishing-date-options'; +import {notify} from 'core/notify/notify'; import ng from 'core/services/ng'; +import {gettext} from 'core/utils'; +import {flatMap, trim} from 'lodash'; +import {IArticle, IDangerousArticlePatchingOptions, IDesk, IStage, onPublishMiddlewareResult} from 'superdesk-api'; +import {duplicateItems} from './article-duplicate'; +import {fetchItems, fetchItemsToCurrentDesk} from './article-fetch'; +import {patchArticle} from './article-patch'; +import {sendItems} from './article-send'; +import {authoringApiCommon} from 'apps/authoring-bridge/authoring-api-common'; +type IArticleActionType = string | 'publish' | 'edit'; const isLocked = (_article: IArticle) => _article.lock_session != null; const isLockedInCurrentSession = (_article: IArticle) => _article.lock_session === ng.get('session').sessionId; const isLockedInOtherSession = (_article: IArticle) => isLocked(_article) && !isLockedInCurrentSession(_article); const isLockedByCurrentUser = (_article: IArticle) => _article.lock_user === ng.get('session').identity._id; const isLockedByOtherUser = (_article: IArticle) => isLocked(_article) && !isLockedByCurrentUser(_article); -const isPublished = (_article: IArticle) => _article.item_id != null; +const isPublished = (item: IArticle, includeScheduled = true) => + PUBLISHED_STATES.includes(item.state) && + (includeScheduled || item.state !== ITEM_STATE.SCHEDULED); const isArchived = (_article: IArticle) => _article._type === 'archived'; const isPersonal = (_article: IArticle) => _article.task == null || _article.task.desk == null || _article.task.stage == null; +const getPackageItemIds = (item: IArticle): Array => { + const ids: Array = []; + + item.groups.forEach((group) => { + if (group.id !== 'root') { + group.refs?.forEach(({residRef}) => { + ids.push(residRef); + }); + } + }); + + return ids; +}; + +/** + * Test if an item was published, but is not published anymore. + */ +export const isKilled = (item: IArticle) => KILLED_STATES.includes(item.state); + +export const isIngested = (item: IArticle) => + item.state === ITEM_STATE.INGESTED; + +function canPublish(item: IArticle): boolean { + const deskId = item?.task?.desk; + + if (deskId == null) { + return false; + } + + const desk = sdApi.desks.getAllDesks().get(deskId); + + if (desk.desk_type === 'authoring' && appConfig?.features?.noPublishOnAuthoringDesk === true) { + return false; + } + + if (sdApi.user.hasPrivilege('publish') !== true) { + return false; + } + + return true; +} + +/** + * Does not prompt for confirmation + */ +function doSpike(item: IArticle) { + return httpRequestJsonLocal({ + method: 'PATCH', + path: `/archive/spike/${item._id}`, + payload: { + state: 'spiked', + }, + headers: { + 'If-Match': item._etag, + }, + }).then(() => { + const $location = ng.get('$location'); + + if ($location.search()._id === item._id) { + $location.search('_id', null); + } + + if (applicationState.articleInEditMode === item._id) { + ng.get('authoringWorkspace').close(); + } + }); +} + +function doUnspike(item: IArticle, deskId: IDesk['_id'], stageId: IStage['_id']): Promise { + return httpRequestJsonLocal({ + method: 'PATCH', + path: `/archive/unspike/${item._id}`, + payload: { + task: { + desk: deskId, + stage: stageId, + }, + }, + headers: { + 'If-Match': item._etag, + }, + }).then(() => { + const $location = ng.get('$location'); + + if ($location.search()._id === item._id) { + $location.search('_id', null); + } + + if (applicationState.articleInEditMode === item._id) { + ng.get('authoringWorkspace').close(); + } + }); +} + +function lock(itemId: IArticle['_id']): Promise { + return httpRequestJsonLocal({ + method: 'POST', + path: `/archive/${itemId}/lock`, + payload: { + lock_action: 'edit', + }, + }); +} + +function unlock(itemId: IArticle['_id']): Promise { + return httpRequestJsonLocal({ + method: 'POST', + path: `/archive/${itemId}/unlock`, + payload: {}, + }); +} + +/** + * Item must be on a stage already. + * i.e. can't be in personal space. + */ +function sendItemToNextStage(item: IArticle): Promise { + if (sdApi.article.isPersonal(item)) { + throw new Error('can not send personal item to next stage'); + } + + const deskId = item.task.desk; + const stageId = item.task.stage; + const deskStages = sdApi.desks.getDeskStages(deskId).toArray(); + const currentStage = deskStages.find(({_id}) => _id === stageId); + const currentStageIndex = deskStages.indexOf(currentStage); + const nextStageIndex = currentStageIndex === deskStages.length - 1 ? 0 : currentStageIndex + 1; + + return sdApi.article.sendItems( + [item], + { + type: 'desk', + desk: deskId, + stage: deskStages[nextStageIndex]._id, + }, + ).then(() => undefined); +} + +function createNewUsingDeskTemplate(): void { + const desk = sdApi.desks.getDeskById(sdApi.desks.getActiveDeskId()); + + sdApi.templates.getById(desk.default_content_template).then((template) => { + ng.get('content') + .createItemFromTemplate(template, false) + .then((item) => { + openArticle(item._id, 'edit'); + }); + }); +} + +/** + * Checks if associations is with rewrite_of item then open then modal to add associations. + * The user has options to add associated media to the current item and review the media change + * or publish the current item without media. + * User will be prompted in following scenarios: + * 1. Edit feature image and confirm media update is enabled. + * 2. Once item is published then no confirmation. + * 3. If current item is update and updated story has associations + */ +function checkMediaAssociatedToUpdate( + item: IArticle, + action: string, + autosave: (item: IArticle) => void, +): Promise { + if (!appConfig.features?.confirmMediaOnUpdate + || !appConfig.features?.editFeaturedImage + || !item.rewrite_of + || ['kill', 'correct', 'takedown'].includes(action) + || item.associations?.featuremedia + ) { + return Promise.resolve(true); + } + + return ng.get('api').find('archive', item.rewrite_of) + .then((rewriteOfItem) => { + if (rewriteOfItem?.associations?.featuremedia) { + return ng.get('confirm').confirmFeatureMedia(rewriteOfItem); + } + + return true; + }) + .then((result) => { + if (result?.associations) { + item.associations = result.associations; + autosave(item); + return false; + } + + return true; + }); +} + +function notifyPreconditionFailed($scope: any) { + notify.error(gettext('Item has changed since it was opened. ' + + 'Please close and reopen the item to continue. ' + + 'Regrettably, your changes cannot be saved.')); + $scope._editable = false; + $scope.dirty = false; +} + +interface IScope { + item?: IArticle; + error?: {}; + autosave?: (item: IArticle) => void; + dirty?: boolean; + $applyAsync?: () => void; + origItem?: IArticle; +} + +function publishItem( + orig: IArticle, + item: IArticle, + action: IArticleActionType = 'publish', + onError?: (error: IPublishingError) => void, +): Promise { + const scope: IScope = {}; + + return publishItem_legacy(orig, item, scope, action, onError) + .then((published) => published ? scope.item : published); +} + +function canPublishOnDesk(deskType: string): boolean { + return !(deskType === 'authoring' && appConfig.features.noPublishOnAuthoringDesk) && + ng.get('privileges').userHasPrivileges({publish: 1}); +} + +function showPublishAndContinue(item: IArticle, dirty: boolean): boolean { + return appConfig.features?.customAuthoringTopbar?.publishAndContinue + && sdApi.navigation.isPersonalSpace() + && canPublishOnDesk(sdApi.desks.getDeskById(sdApi.desks.getCurrentDeskId()).desk_type) + && authoringApiCommon.checkShortcutButtonAvailability(item, dirty, sdApi.navigation.isPersonalSpace()); +} + +function showCloseAndContinue(item: IArticle, dirty: boolean): boolean { + return appConfig.features?.customAuthoringTopbar?.closeAndContinue + && !sdApi.navigation.isPersonalSpace() + && authoringApiCommon.checkShortcutButtonAvailability(item, dirty); +} + +function publishItem_legacy( + orig: IArticle, + item: IArticle, + scope: IScope, + action: string | 'publish' | 'edit' = 'publish', + onError?: (error: IPublishingError) => void, +): Promise { + let warnings: Array<{text: string}> = []; + const initialValue: Promise = Promise.resolve({}); + + scope.error = {}; + + return flatMap( + Object.values(extensions).map(({activationResult}) => activationResult), + (activationResult) => activationResult.contributions?.entities?.article?.onPublish ?? [], + ).reduce((current, next) => { + return current.then((result) => { + if ((result?.warnings?.length ?? 0) > 0) { + warnings = warnings.concat(result.warnings); + } + + return next(Object.assign({ + _id: orig._id, + type: orig.type, + }, item)); + }); + }, initialValue) + .then((result) => { + if ((result?.warnings?.length ?? 0) > 0) { + warnings = warnings.concat(result.warnings); + } + + return result; + }) + .then(() => checkMediaAssociatedToUpdate(item, action, scope.autosave)) + .then((result) => (result && warnings.length < 1 + ? ng.get('authoring').publish(orig, item, action) + : Promise.reject(false) + )) + .then((response: IArticle) => { + notify.success(gettext('Item published.')); + scope.item = response; + scope.dirty = false; + ng.get('authoringWorkspace').close(true); + + return true; + }) + .catch((response) => { + const issues = response.data._issues; + const errors = issues?.['validator exception']; + + if (errors != null) { + const modifiedErrors = errors.replace(/\[/g, '').replace(/\]/g, '').split(','); + + modifiedErrors.forEach((error) => { + const message = trim(error, '\' '); + // the message format is 'Field error text' (contains ') + const field = message.split(' ')[0]; + + scope.error[field.toLocaleLowerCase()] = true; + notify.error(message); + }); + + if (issues.fields) { + Object.assign(scope.error, issues.fields); + } + + onError?.({ + fields: scope.error, + kind: 'publishing-error', + }); + + scope.$applyAsync?.(); // make $scope.error changes visible + + if (errors.indexOf('9007') >= 0 || errors.indexOf('9009') >= 0) { + ng.get('authoring').open(item._id, true).then((res) => { + scope.origItem = res; + scope.dirty = false; + scope.item = copyJson(scope.origItem); + }); + } + } else if (issues?.unique_name?.unique) { + notify.error(gettext('Error: Unique Name is not unique.')); + } else if (response && response.status === 412) { + notifyPreconditionFailed(scope); + } else if (warnings.length > 0) { + warnings.forEach((warning) => notify.error(warning.text)); + } + + return Promise.reject(false); + }); +} + +/** + * Gets opened items from your workspace. + */ +function getWorkQueueItems(): Array { + return ng.get('workqueue').items; +} + +function get(id: IArticle['_id']): Promise { + return dataApi.findOne('archive', id); +} + +function isEditable(_article: IArticle): boolean { + const itemState: ITEM_STATE = _article.state; + const authoring = ng.get('authoring'); + + switch (itemState) { + case ITEM_STATE.DRAFT: + case ITEM_STATE.CORRECTION: + case ITEM_STATE.SUBMITTED: + case ITEM_STATE.IN_PROGRESS: + case ITEM_STATE.ROUTED: + case ITEM_STATE.FETCHED: + case ITEM_STATE.UNPUBLISHED: + return authoring.itemActions(_article).edit === true; + case ITEM_STATE.INGESTED: + case ITEM_STATE.SPIKED: + case ITEM_STATE.SCHEDULED: + case ITEM_STATE.PUBLISHED: + case ITEM_STATE.CORRECTED: + case ITEM_STATE.BEING_CORRECTED: + case ITEM_STATE.KILLED: + case ITEM_STATE.RECALLED: + return false; + default: + assertNever(itemState); + } +} + +function rewrite(item: IArticle): void { + return ng.get('authoring').rewrite(item); +} interface IArticleApi { + get(id: IArticle['_id']): Promise; isLocked(article: IArticle): boolean; + isEditable(article: IArticle): boolean; isLockedInCurrentSession(article: IArticle): boolean; isLockedInOtherSession(article: IArticle): boolean; isLockedByCurrentUser(article: IArticle): boolean; isLockedByOtherUser(article: IArticle): boolean; isArchived(article: IArticle): boolean; - isPublished(article: IArticle): boolean; + + /** + * @param includeScheduled defaults to true + */ + isPublished(article: IArticle, includeScheduled?: boolean): boolean; + + isKilled(article: IArticle): boolean; + isIngested(article: IArticle): boolean; isPersonal(article: IArticle): boolean; + getPackageItemIds(item: IArticle): Array; + patch( + article: IArticle, + patch: Partial, + dangerousOptions?: IDangerousArticlePatchingOptions, + ): Promise; + doSpike(item: IArticle): Promise; + doUnspike(item: IArticle, deskId: IDesk['_id'], stageId: IStage['_id']): Promise; + + fetchItems( + items: Array, + selectedDestination: ISendToDestinationDesk, + ): Promise>; + + fetchItemsToCurrentDesk(items: Array): Promise>; + + /** + * Promise may be rejected by middleware. + * Returns patches, not whole items. + */ + sendItems( + items: Array, + selectedDestination: ISendToDestination, + sendPackageItems?: boolean, + publishingDateOptions?: IPublishingDateOptions, + ): Promise>>; + + sendItemToNextStage(item: IArticle): Promise; + + duplicateItems(items: Array, destination: ISendToDestination): Promise>; + + canPublish(item: IArticle): boolean; + + lock(itemId: IArticle['_id']): Promise; + unlock(itemId: IArticle['_id']): Promise; + + createNewUsingDeskTemplate(): void; + getWorkQueueItems(): Array; + rewrite(item: IArticle): void; + canPublishOnDesk(deskType: string): boolean; + showCloseAndContinue(item: IArticle, dirty: boolean): boolean; + showPublishAndContinue(item: IArticle, dirty: boolean): boolean; + publishItem_legacy(orig: IArticle, item: IArticle, $scope: any, action?: string): Promise; + + // Instead of passing a fake scope from React + // every time to the publishItem_legacy we can use this function which + // creates a fake scope for us. + publishItem( + orig: IArticle, + item: IArticle, + action?: string, + + // onError is optional in this function and in `publishItem_legacy` since when you're calling + // it from React you want to pass only it to handle certain errors and apply them to the scope + // but not the whole scope but from Angular you already have access to the full scope so you + // won't need to pass onError + onError?: (error: IPublishingError) => void, + ): Promise; } export const article: IArticleApi = { + rewrite, isLocked, + isEditable, isLockedInCurrentSession, isLockedInOtherSession, isLockedByCurrentUser, isLockedByOtherUser, isArchived, isPublished, + isKilled, + isIngested, isPersonal, + getPackageItemIds, + patch: patchArticle, + doSpike, + doUnspike, + fetchItems, + fetchItemsToCurrentDesk, + sendItems, + sendItemToNextStage, + duplicateItems, + canPublish, + lock, + unlock, + createNewUsingDeskTemplate, + getWorkQueueItems, + get, + canPublishOnDesk, + showPublishAndContinue, + showCloseAndContinue, + publishItem_legacy, + publishItem, }; diff --git a/scripts/api/config.tsx b/scripts/api/config.tsx new file mode 100644 index 0000000000..80a6ca5c46 --- /dev/null +++ b/scripts/api/config.tsx @@ -0,0 +1,9 @@ +import ng from 'core/services/ng'; + +function featureEnabled(name: string): boolean { + return ng.get('features')[name] != null; +} + +export const config = { + featureEnabled, +}; diff --git a/scripts/api/desks.ts b/scripts/api/desks.ts index f3fb6c1ec4..36f516a7d5 100644 --- a/scripts/api/desks.ts +++ b/scripts/api/desks.ts @@ -1,10 +1,15 @@ -import {IDesk} from 'superdesk-api'; +import {IDesk, IStage} from 'superdesk-api'; import ng from 'core/services/ng'; +import {OrderedMap} from 'immutable'; function getActiveDeskId(): IDesk['_id'] | null { return ng.get('desks').activeDeskId; } +function getCurrentDeskId(): IDesk['_id'] | null { + return ng.get('desks').getCurrentDeskId(); +} + function waitTilReady(): Promise { return new Promise((resolve) => { ng.get('desks') @@ -16,12 +21,51 @@ function waitTilReady(): Promise { }); } +function getAllDesks(): OrderedMap { + let desksMap: OrderedMap = OrderedMap(); + + for (const desk of ng.get('desks').desks._items) { + desksMap = desksMap.set(desk._id, desk); + } + + return desksMap; +} + +function getCurrentUserDesks(): Array { + return ng.get('desks').userDesks; +} + +function getDeskStages(deskId: IDesk['_id']): OrderedMap { + let stagesMap: OrderedMap = OrderedMap(); + + for (const stage of ng.get('desks').deskStages[deskId]) { + stagesMap = stagesMap.set(stage._id, stage); + } + + return stagesMap; +} + +function getDeskById(id: IDesk['_id']): IDesk { + return getAllDesks().get(id); +} + interface IDesksApi { + /** Desk is considered active if it is being viewed in monitoring at the moment */ getActiveDeskId(): IDesk['_id'] | null; + getCurrentDeskId(): IDesk['_id'] | null; waitTilReady(): Promise; + getAllDesks(): OrderedMap; + getDeskById(id: IDesk['_id']): IDesk ; + getDeskStages(deskId: IDesk['_id']): OrderedMap; + getCurrentUserDesks(): Array; // desks that current user has access to } export const desks: IDesksApi = { getActiveDeskId, + getCurrentDeskId, waitTilReady, + getAllDesks, + getDeskById, + getDeskStages, + getCurrentUserDesks, }; diff --git a/scripts/api/highlights.ts b/scripts/api/highlights.ts new file mode 100644 index 0000000000..2dd57e1234 --- /dev/null +++ b/scripts/api/highlights.ts @@ -0,0 +1,30 @@ +import {httpRequestJsonLocal} from 'core/helpers/network'; +import {IHighlightResponse} from 'superdesk-api'; + +function markItem(highlightIds: Array, itemId: string): Promise { + return httpRequestJsonLocal({ + payload: { + highlights: highlightIds, + marked_item: itemId, + }, + method: 'POST', + path: '/marked_for_highlights', + }); +} + +function fetchHighlights(): Promise { + return httpRequestJsonLocal({ + method: 'GET', + path: '/highlights', + }); +} + +interface IHighlightsApi { + markItem(highlighIds: Array, itemId: string): Promise; + fetchHighlights(): Promise; +} + +export const highlights: IHighlightsApi = { + markItem, + fetchHighlights, +}; diff --git a/scripts/api/index.ts b/scripts/api/index.ts index 83908e9bef..c7b8e31fc2 100644 --- a/scripts/api/index.ts +++ b/scripts/api/index.ts @@ -1,6 +1,15 @@ +import {highlights} from './highlights'; import {article} from './article'; +import {config} from './config'; import {desks} from './desks'; import {ingest} from './ingest'; +import {localStorage} from './local-storage'; +import {navigation} from './navigation'; +import {preferences} from './preferences'; +import {templates} from './templates'; +import {time} from './time'; +import {user} from './user'; +import {vocabularies} from './vocabularies'; /** * This is core API, not extensions API. @@ -9,6 +18,15 @@ import {ingest} from './ingest'; */ export const sdApi = { article, + config, desks, ingest, + localStorage, + navigation, + preferences, + templates, + time, + user, + vocabularies, + highlights, }; diff --git a/scripts/api/local-storage.ts b/scripts/api/local-storage.ts new file mode 100644 index 0000000000..bc27f7361e --- /dev/null +++ b/scripts/api/local-storage.ts @@ -0,0 +1,24 @@ +import ng from 'core/services/ng'; + +function getItem(key: string) { + return ng.get('storage').getItem(key); +} + +function setItem(key: string, data: any) { + return ng.get('storage').setItem(key, data); +} + +function removeItem(key: string) { + return ng.get('storage').removeItem(key); +} + +function clear() { + return ng.get('storage').clear(); +} + +export const localStorage = { + getItem, + setItem, + removeItem, + clear, +}; diff --git a/scripts/api/navigation.ts b/scripts/api/navigation.ts new file mode 100644 index 0000000000..80b08ab3a5 --- /dev/null +++ b/scripts/api/navigation.ts @@ -0,0 +1,30 @@ +import ng from 'core/services/ng'; + +function getPath(): string { + return ng.get('$location').path(); +} + +function currentPathStartsWith( + sections: Array, // example: ['workspace', 'personal'] or ['settings', 'content-profiles'] +): boolean { + const path = getPath(); + const pathSections = path.split('/').slice(1); // remove "/" at the start + + for (let i = 0; i < sections.length; i++) { + if (sections[i] !== pathSections[i]) { + return false; + } + } + + return true; +} + +function isPersonalSpace(): boolean { + return !(ng.get('$location').path() === '/workspace/personal'); +} + +export const navigation = { + getPath, + currentPathStartsWith, + isPersonalSpace, +}; diff --git a/scripts/api/preferences.ts b/scripts/api/preferences.ts new file mode 100644 index 0000000000..2b99dfa8a8 --- /dev/null +++ b/scripts/api/preferences.ts @@ -0,0 +1,18 @@ +import ng from 'core/services/ng'; + +function get(key: string) { + const preferencesService = ng.get('preferencesService'); + + return preferencesService.getSync(key); +} + +function update(key: string, value: any): void { + const preferencesService = ng.get('preferencesService'); + + preferencesService.update({[key]: value}, key); +} + +export const preferences = { + get, + update, +}; diff --git a/scripts/api/templates.ts b/scripts/api/templates.ts new file mode 100644 index 0000000000..8d0041524a --- /dev/null +++ b/scripts/api/templates.ts @@ -0,0 +1,111 @@ +import { + applyMiddleware, + canEdit, + cleanData, + prepareData, + willCreateNew, +} from 'apps/authoring-react/toolbar/template-helpers'; +import {httpRequestJsonLocal} from 'core/helpers/network'; +import ng from 'core/services/ng'; +import {clone} from 'lodash'; +import {IArticle, IDesk, ITemplate} from 'superdesk-api'; + +function getById(id: ITemplate['_id']): Promise { + return httpRequestJsonLocal({ + method: 'GET', + path: `/content_templates/${id}`, + }); +} + +function createTemplate(payload) { + return httpRequestJsonLocal({ + method: 'POST', + path: '/content_templates', + payload, + }); +} + +function updateTemplate(payload, template: ITemplate) { + return httpRequestJsonLocal({ + method: 'PATCH', + path: `/content_templates/${template._id}`, + payload, + headers: {'If-Match': template._etag}, + }); +} + +/** + * Creates or updates a template. If the article has an existing template it will be updated. + * + * @templateName - template name from the form input + * @selectedDeskId - deskId selected in the form + */ +function createTemplateFromArticle( + // The new template will be based on this article + sourceArticle: IArticle, + templateName: string, + selectedDeskId: IDesk['_id'] | null, +): Promise { + return getById(sourceArticle.template).then((resultTemplate) => { + const data = prepareData(resultTemplate); + const deskId = selectedDeskId || data.desk; + const templateArticle = data.template; + + // Clean the article from fields not usable for template creation + const item: IArticle = clone(ng.get('templates').pickItemData(sourceArticle)); + const userId = ng.get('session').identity._id; + + return applyMiddleware(item).then((itemAfterMiddleware) => { + const newTemplate: Partial = { + template_name: templateName, + template_type: 'create', + template_desks: selectedDeskId != null ? [deskId] : null, + is_public: templateArticle.is_public, + data: itemAfterMiddleware, + }; + + let templateTemp: Partial = templateArticle != null ? templateArticle : newTemplate; + let diff = templateArticle != null ? newTemplate : null; + + if (willCreateNew(templateArticle, templateName, selectedDeskId != null)) { + templateTemp = newTemplate; + diff = null; + + if (!canEdit(templateArticle, selectedDeskId != null)) { + templateTemp.is_public = false; + templateTemp.user = userId; + templateTemp.template_desks = null; + } + } + + const hasLinks = templateTemp._links != null; + const payload: Partial = diff != null ? cleanData(diff) : cleanData(templateTemp); + + // if the template is made private, set the current user as template owner + if (templateArticle.is_public && (diff?.is_public === false || templateTemp.is_public === false)) { + payload.user = userId; + } + + const requestPayload: Partial = { + ...payload, + data: item, + }; + + return (hasLinks + ? updateTemplate(requestPayload, resultTemplate) + : createTemplate(requestPayload) + ) + .then((_data) => { + return _data; + }, (response) => { + return Promise.reject(response); + }); + }); + }); +} + +export const templates = { + getById, + createTemplateFromArticle, + prepareData, +}; diff --git a/scripts/api/time.ts b/scripts/api/time.ts new file mode 100644 index 0000000000..a993d2eebd --- /dev/null +++ b/scripts/api/time.ts @@ -0,0 +1,19 @@ +import {getTimezoneLabels} from 'apps/dashboard/world-clock/timezones-all-labels'; +import {OrderedMap} from 'immutable'; + +/** + * Returns translated labels of time zones indexed by time zone IDs + */ +function getTimeZones(): OrderedMap { + let result = OrderedMap(); + + for (const [id, label] of Object.entries(getTimezoneLabels())) { + result = result.set(id, label); + } + + return result; +} + +export const time = { + getTimeZones, +}; diff --git a/scripts/api/user.ts b/scripts/api/user.ts new file mode 100644 index 0000000000..7f35352f55 --- /dev/null +++ b/scripts/api/user.ts @@ -0,0 +1,18 @@ +import ng from 'core/services/ng'; + +function hasPrivilege(privilege: string): boolean { + const privileges = ng.get('privileges'); + + return privileges.userHasPrivileges({[privilege]: 1}); +} + +function isLoggedIn() { + const session = ng.get('session'); + + return session?.identity?._id != null; +} + +export const user = { + hasPrivilege, + isLoggedIn, +}; diff --git a/scripts/api/vocabularies.ts b/scripts/api/vocabularies.ts new file mode 100644 index 0000000000..90bd033773 --- /dev/null +++ b/scripts/api/vocabularies.ts @@ -0,0 +1,109 @@ +import {OrderedMap} from 'immutable'; +import ng from 'core/services/ng'; +import {IArticle, IVocabulary, IVocabularyItem} from 'superdesk-api'; +import {getVocabularyItemNameTranslated} from 'core/utils'; + +function getAll(): OrderedMap { + return OrderedMap( + ng.get('vocabularies').getAllVocabulariesSync().map( + (vocabulary) => [vocabulary._id, vocabulary], + ), + ); +} + +function isCustomFieldVocabulary(vocabulary: IVocabulary): boolean { + return vocabulary.field_type != null || vocabulary.custom_field_type != null; +} + +function getVocabularyItemLabel(term: IVocabularyItem, item: IArticle): string { + if (!term) { + return 'None'; + } + + // Item can be anything here. It might be an article object or search filters object + // depending where the function is called from. + // It's checked if language is a string in order not to confuse it when language + // is an array when called from global search filters. + const language = typeof item.language === 'string' ? item.language : undefined; + + return getVocabularyItemNameTranslated(term, language); +} + +const vocabularyItemsToString = ( + array: Array, + propertyName?: keyof IVocabularyItem, + schemeName?: string, +): string => + getVocabularyItemsByPropertyName(array, propertyName, schemeName).join(', '); + +const getVocabularyItemsByPropertyName = ( + array: Array, + propertyName?: keyof IVocabularyItem, + schemeName?: string, +): Array => { + let subjectMerged = []; + + array.forEach((item) => { + const value = propertyName == null ? item : item[propertyName]; + + if (value) { + subjectMerged.push(value); + + if ((schemeName?.length ?? 0) < 1 && item.scheme !== schemeName) { + subjectMerged.pop(); + } + } + }); + + return subjectMerged; +}; + +const getVocabularyItemsPreview = ( + array: Array, + propertyName?: keyof IVocabularyItem, + schemeName?: string, + returnArray?: boolean, +): Array | string => { + if (returnArray) { + return getVocabularyItemsByPropertyName(array, propertyName, schemeName); + } else { + return vocabularyItemsToString(array, propertyName, schemeName); + } +}; + +/** + * Selection vocabularies may be configured to be included in content profiles. + */ +function isSelectionVocabulary(vocabulary: IVocabulary): boolean { + return !isCustomFieldVocabulary(vocabulary) && ( + vocabulary.selection_type === 'multi selection' + || vocabulary.selection_type === 'single selection' + ); +} + +interface IVocabulariesApi { + getAll: () => OrderedMap; + isCustomFieldVocabulary: (vocabulary: IVocabulary) => boolean; + isSelectionVocabulary: (vocabulary: IVocabulary) => boolean; + getVocabularyItemLabel: (term: IVocabularyItem, item: IArticle) => string; + getVocabularyItemsPreview: ( + array: Array, + propertyName?: keyof IVocabularyItem, + schemeName?: string, + returnArray?: boolean, + ) => Array | string; + vocabularyItemsToString: ( + array: Array, + propertyName?: keyof IVocabularyItem, + schemeName?: string, + ) => string; +} + +export const vocabularies: IVocabulariesApi = { + getAll, + isCustomFieldVocabulary, + isSelectionVocabulary, + getVocabularyItemLabel, + getVocabularyItemsPreview, + vocabularyItemsToString, +}; diff --git a/scripts/appConfig.ts b/scripts/appConfig.ts index 74482f6b94..7a5a2132e4 100644 --- a/scripts/appConfig.ts +++ b/scripts/appConfig.ts @@ -41,3 +41,26 @@ export function getUserInterfaceLanguage() { return 'en'; } } + +export const debugInfo = { + translationsLoaded: false, +}; + +export let authoringReactEnabledUserSelection = (JSON.parse(localStorage.getItem('auth-react') ?? 'false') as boolean); + +export function toggleAuthoringReact(enabled: boolean) { + localStorage.setItem('auth-react', JSON.stringify(enabled)); + + authoringReactEnabledUserSelection = enabled; + return authoringReactEnabledUserSelection; +} +/** + * Authoring react has to be enabled in the broadcasting + * module regardless of the user selection. + * */ +export let authoringReactViewEnabled = authoringReactEnabledUserSelection; +export const uiFrameworkAuthoringPanelTest = false; + +export function setAuthoringReact(enabled: boolean) { + authoringReactViewEnabled = enabled; +} diff --git a/scripts/apps/archive/controllers/UploadController.ts b/scripts/apps/archive/controllers/UploadController.ts index 8f59b4ec67..35e90452dd 100644 --- a/scripts/apps/archive/controllers/UploadController.ts +++ b/scripts/apps/archive/controllers/UploadController.ts @@ -7,7 +7,7 @@ import {extensions} from 'appConfig'; import {IPTCMetadata, IUser, IArticle} from 'superdesk-api'; import {appConfig} from 'appConfig'; import {fileUploadErrorModal} from './file-upload-error-modal'; -import {showModal} from 'core/services/modalService'; +import {showModal} from '@superdesk/common'; const isNotEmptyString = (value: any) => value != null && value !== ''; @@ -332,10 +332,24 @@ export function UploadController( }); if (uploadOfDisallowedFileTypesAttempted) { - const message = gettext('Only the following files are allowed: ') - + ($scope.allowPicture ? gettext('image') : '') - + ($scope.allowVideo ? ', ' + gettext('video') : '') - + ($scope.allowAudio ? ', ' + gettext('audio') : ''); + const allowedTypes = []; + + if ($scope.allowPicture) { + allowedTypes.push(gettext('image')); + } + + if ($scope.allowVideo) { + allowedTypes.push(gettext('video')); + } + + if ($scope.allowAudio) { + allowedTypes.push(gettext('audio')); + } + + const message = gettext( + 'Only the following files are allowed: {{fileTypes}}', + {fileTypes: allowedTypes.join(', ')}, + ); notify.error(message); } diff --git a/scripts/apps/archive/directives/HtmlPreview.ts b/scripts/apps/archive/directives/HtmlPreview.ts index e36678ba92..51e4e5fdcc 100644 --- a/scripts/apps/archive/directives/HtmlPreview.ts +++ b/scripts/apps/archive/directives/HtmlPreview.ts @@ -2,6 +2,7 @@ import {getAnnotationsFromItem} from 'core/editor3/helpers/editor3CustomData'; import {META_FIELD_NAME} from 'core/editor3/helpers/fieldsMeta'; import ng from 'core/services/ng'; import {gettext} from 'core/utils'; +import {IArticle} from 'superdesk-api'; function getAnnotationTypesAsync(scope) { ng.get('metadata').initialize() @@ -16,7 +17,15 @@ function getAnnotationTypesAsync(scope) { }); } -function getAllAnnotations(item) { +interface IAnotationData { + body: string; + id: number; + index: number; + styleName: string; + type: string; +} + +export function getAllAnnotations(item: IArticle): Array { const annotations = []; for (const field in item[META_FIELD_NAME]) { diff --git a/scripts/apps/archive/directives/ItemPreviewContainer.ts b/scripts/apps/archive/directives/ItemPreviewContainer.ts deleted file mode 100644 index 457cec4824..0000000000 --- a/scripts/apps/archive/directives/ItemPreviewContainer.ts +++ /dev/null @@ -1,43 +0,0 @@ -import {IArticle} from 'superdesk-api'; -import {dispatchCustomEvent} from 'core/get-superdesk-api-implementation'; - -let itemInPreviewMode: IArticle | null = null; - -export function ItemPreviewContainer() { - return { - template: '
', - scope: {}, - link: function(scope) { - scope.item = null; - - scope.$on('intent:preview:item', (event, intent) => { - if (itemInPreviewMode != null) { - dispatchCustomEvent('articlePreviewEnd', itemInPreviewMode); - - itemInPreviewMode = null; - } - - if (intent.data != null) { - itemInPreviewMode = intent.data; - - dispatchCustomEvent('articlePreviewStart', itemInPreviewMode); - } - - scope.item = intent.data; - }); - - /** - * Close lightbox - */ - scope.close = function() { - scope.item = null; - - if (itemInPreviewMode != null) { - dispatchCustomEvent('articlePreviewEnd', itemInPreviewMode); - - itemInPreviewMode = null; - } - }; - }, - }; -} diff --git a/scripts/apps/archive/directives/MediaPreview.ts b/scripts/apps/archive/directives/MediaPreview.ts index 37f423bb89..348f99b61a 100644 --- a/scripts/apps/archive/directives/MediaPreview.ts +++ b/scripts/apps/archive/directives/MediaPreview.ts @@ -1,6 +1,6 @@ import _ from 'lodash'; import {checkRenditions, getAssociationsByFieldId} from 'apps/authoring/authoring/controllers/AssociationController'; -import {IArticleField} from 'superdesk-api'; +import {IVocabulary} from 'superdesk-api'; import {appConfig} from 'appConfig'; import {gettext} from 'core/utils'; @@ -30,7 +30,7 @@ export function MediaPreview(api, $rootScope, desks, superdesk, content, storage const setSubjectPreviewFields = () => { scope.subjectPreviewFields = content.previewFields(scope.editor, scope.fields) - .filter((field: IArticleField) => field.field_type == null); + .filter((field: IVocabulary) => field.field_type == null); }; scope.isCorrectionWorkflowEnabled = appConfig?.corrections_workflow; diff --git a/scripts/apps/archive/directives/MediaView.ts b/scripts/apps/archive/directives/MediaView.ts deleted file mode 100644 index e4a6c04c9b..0000000000 --- a/scripts/apps/archive/directives/MediaView.ts +++ /dev/null @@ -1,102 +0,0 @@ -import _ from 'lodash'; - -MediaView.$inject = ['keyboardManager', 'packages']; - -export function MediaView(keyboardManager, packages) { - return { - templateUrl: 'scripts/apps/archive/views/media-view.html', - scope: { - items: '=', - item: '=', - close: '&', - }, - link: function(scope, elem) { - var packageStack = []; - - scope.singleItem = null; - scope.packageItem = null; - - scope.prevEnabled = true; - scope.nextEnabled = true; - - var getIndex = function(item) { - return _.findIndex(scope.items, {_id: item._id}); - }; - - var setItem = function(item) { - resetStack(); - scope.item = item; - scope.openItem(item); - var index = getIndex(scope.item); - - scope.prevEnabled = index > -1 && !!scope.items[index - 1]; - scope.nextEnabled = index > -1 && !!scope.items[index + 1]; - }; - - scope.prev = function() { - var index = getIndex(scope.item); - - if (index > 0) { - setItem(scope.items[index - 1]); - } - }; - scope.next = function() { - var index = getIndex(scope.item); - - if (index !== -1 && index < scope.items.length - 1) { - setItem(scope.items[index + 1]); - } - }; - - keyboardManager.push('left', scope.prev); - keyboardManager.push('right', scope.next); - keyboardManager.bind('esc', () => { - scope.close(); - }, {global: true}); - scope.$on('$destroy', () => { - keyboardManager.pop('left'); - keyboardManager.pop('right'); - keyboardManager.unbind('esc'); - }); - - scope.setPackageSingle = function(packageItem) { - packages.fetchItem(packageItem).then((item) => { - scope.openItem(item); - }); - }; - - scope.openItem = function(item) { - if (item.type === 'composite') { - packageStack.push(item); - pickPackageItem(); - } - scope.setSingleItem(item); - }; - - scope.setSingleItem = function(item) { - scope.singleItem = item; - }; - - scope.nested = function() { - return packageStack.length > 1; - }; - - scope.previousPackage = function() { - _.remove(packageStack, _.last(packageStack)); - pickPackageItem(); - scope.setSingleItem(scope.packageItem); - }; - - var pickPackageItem = function() { - scope.packageItem = _.last(packageStack) || null; - }; - - var resetStack = function() { - packageStack = []; - scope.packageItem = null; - }; - - setItem(scope.item); - }, - }; -} diff --git a/scripts/apps/archive/directives/RelatedItemsPreview.ts b/scripts/apps/archive/directives/RelatedItemsPreview.ts index 35e0d4e378..d08e28abe3 100644 --- a/scripts/apps/archive/directives/RelatedItemsPreview.ts +++ b/scripts/apps/archive/directives/RelatedItemsPreview.ts @@ -1,4 +1,4 @@ -import {IArticle, IArticleField, IRendition} from 'superdesk-api'; +import {IArticle, IVocabulary, IRendition} from 'superdesk-api'; import {gettext} from 'core/utils'; import {getThumbnailForItem} from 'core/helpers/item'; import {throttle} from 'lodash'; @@ -16,7 +16,7 @@ import {throttle} from 'lodash'; interface IScope extends ng.IScope { item: IArticle; - field: IArticleField; + field: IVocabulary; preview: boolean; loading: boolean; relatedItems: Array; diff --git a/scripts/apps/archive/directives/SingleItem.ts b/scripts/apps/archive/directives/SingleItem.ts deleted file mode 100644 index e855d7a14e..0000000000 --- a/scripts/apps/archive/directives/SingleItem.ts +++ /dev/null @@ -1,10 +0,0 @@ -export function SingleItem() { - return { - templateUrl: 'scripts/apps/archive/views/single-item-preview.html', - scope: { - item: '=', - contents: '=', - setitem: '&', - }, - }; -} diff --git a/scripts/apps/archive/directives/index.ts b/scripts/apps/archive/directives/index.ts index 11e5cdf56d..4fab563ca0 100644 --- a/scripts/apps/archive/directives/index.ts +++ b/scripts/apps/archive/directives/index.ts @@ -2,12 +2,9 @@ export {ItemLock} from './ItemLock'; export {ItemState} from './ItemState'; export {InlineMeta} from './InlineMeta'; export {MediaPreview, MediaPreviewWidget} from './MediaPreview'; -export {ItemPreviewContainer} from './ItemPreviewContainer'; -export {MediaView} from './MediaView'; export {MediaMetadata} from './MediaMetadata'; export {FetchedDesks} from './FetchedDesks'; export {MetaIngest} from './MetaIngest'; -export {SingleItem} from './SingleItem'; export {DraggableItem} from './DraggableItem'; export {ItemCrops} from './ItemCrops'; export {ItemRendition} from './ItemRendition'; diff --git a/scripts/apps/archive/index.tsx b/scripts/apps/archive/index.tsx index d4f5b524a0..f72c55696e 100644 --- a/scripts/apps/archive/index.tsx +++ b/scripts/apps/archive/index.tsx @@ -25,6 +25,9 @@ import * as actions from './actions'; import {RelatedView} from './views/related-view'; import {showUnsavedChangesPrompt, IUnsavedChangesAction} from 'core/ui/components/prompt-for-unsaved-changes'; import {assertNever} from 'core/helpers/typescript-helpers'; +import {httpRequestJsonLocal, httpRequestRawLocal} from 'core/helpers/network'; +import {sdApi} from 'api'; +import {dispatchInternalEvent} from 'core/internal-events'; angular.module('superdesk.apps.archive.directives', [ 'superdesk.core.filters', @@ -37,13 +40,10 @@ angular.module('superdesk.apps.archive.directives', [ .directive('sdInlineMeta', directive.InlineMeta) .directive('sdMediaPreview', directive.MediaPreview) .directive('sdMediaPreviewWidget', directive.MediaPreviewWidget) - .directive('sdItemPreviewContainer', directive.ItemPreviewContainer) - .directive('sdMediaView', directive.MediaView) .directive('sdMediaMetadata', directive.MediaMetadata) .component('sdRelatedView', reactToAngular1(RelatedView, ['relatedItems'], [])) .directive('sdFetchedDesks', directive.FetchedDesks) .directive('sdMetaIngest', directive.MetaIngest) - .directive('sdSingleItem', directive.SingleItem) .directive('sdDraggableItem', directive.DraggableItem) .directive('sdItemCrops', directive.ItemCrops) .directive('sdItemRendition', directive.ItemRendition) @@ -79,8 +79,6 @@ angular.module('superdesk.apps.archive', [ 'superdesk.apps.dashboard.widgets.relatedItem', 'superdesk.apps.workspace.menu', ]) - - .service('spike', svc.SpikeService) .service('multi', svc.MultiService) .service('archiveService', svc.ArchiveService) @@ -133,10 +131,11 @@ angular.module('superdesk.apps.archive', [ label: gettext('Unspike Item'), icon: 'unspike', monitor: true, - controller: ['spike', 'data', '$rootScope', function unspikeActivity(spike, data, $rootScope) { - return spike.unspike(data.item).then((item) => { - $rootScope.$broadcast('item:unspike'); - return item; + controller: ['data', function(data) { + dispatchInternalEvent('interactiveArticleActionStart', { + items: [data.item], + tabs: ['unspike'], + activeTab: 'unspike', }); }], filters: [{action: 'list', type: 'spike'}], @@ -170,8 +169,12 @@ angular.module('superdesk.apps.archive', [ label: gettext('Duplicate To'), icon: 'copy', monitor: true, - controller: ['data', 'send', function(data, send) { - return send.allAs([data.item], 'duplicateTo'); + controller: ['data', function(data) { + dispatchInternalEvent('interactiveArticleActionStart', { + items: [data.item], + tabs: ['duplicate_to'], + activeTab: 'duplicate_to', + }); }], filters: [ {action: 'list', type: 'archive'}, @@ -401,7 +404,6 @@ angular.module('superdesk.apps.archive', [ }]); spikeActivity.$inject = [ - 'spike', 'data', 'modal', '$location', @@ -409,10 +411,11 @@ spikeActivity.$inject = [ 'authoringWorkspace', 'confirm', 'autosave', + '$rootScope', ]; -function spikeActivity(spike, data, modal, $location, multi, - authoringWorkspace: AuthoringWorkspaceService, confirm, autosave) { +function spikeActivity(data, modal, $location, multi, + authoringWorkspace: AuthoringWorkspaceService, confirm, autosave, $rootScope) { // For the sake of keyboard shortcut to work consistently, // if the item is multi-selected, let multibar controller handle its spike if (!data.item || multi.count > 0 && includes(multi.getIds(), data.item._id)) { @@ -433,8 +436,20 @@ function spikeActivity(spike, data, modal, $location, multi, break; case IUnsavedChangesAction.discardChanges: - closePromptFn(); - _spike(); + httpRequestJsonLocal({ + method: 'GET', + path: `/archive_autosave/${data.item._id}`, + }).then((item) => httpRequestRawLocal({ + method: 'DELETE', + path: `/archive_autosave/${item._id}`, + headers: { + 'If-Match': item._etag, + }, + }).then(() => { + $rootScope.$applyAsync(); + closePromptFn(); + _spike(); + })); break; @@ -456,7 +471,7 @@ function spikeActivity(spike, data, modal, $location, multi, function _spike() { if ($location.path() === '/workspace/personal') { return modal.confirm(gettext('Do you want to delete the item permanently?'), gettext('Confirm')) - .then(() => spike.spike(data.item)); + .then(() => sdApi.article.doSpike(data.item)); } const onSpikeMiddlewares @@ -476,7 +491,7 @@ function spikeActivity(spike, data, modal, $location, multi, showSpikeDialog( modal, - () => spike.spike(data.item), + () => sdApi.article.doSpike(data.item), gettext('Are you sure you want to spike the item?'), onSpikeMiddlewares, item, diff --git a/scripts/apps/archive/services/SpikeService.ts b/scripts/apps/archive/services/SpikeService.ts deleted file mode 100644 index ed05749892..0000000000 --- a/scripts/apps/archive/services/SpikeService.ts +++ /dev/null @@ -1,129 +0,0 @@ -import {gettext} from 'core/utils'; -import {AuthoringWorkspaceService} from 'apps/authoring/authoring/services/AuthoringWorkspaceService'; - -/** - * @ngdoc service - * @module superdesk.apps.archive - * @name spike - * - * @requires $location - * @requires api - * @requires notify - * @requires send - * @requires $q - * @requires authoring - * @requires authoringWorkspace - * @requires superdeskFlags - * - * @description Spike Service is responsible for proving item (single and multiple) spike/un-spike functionality - */ - -SpikeService.$inject = ['$location', 'api', 'notify', 'send', '$q', 'authoringWorkspace', 'lock']; -export function SpikeService($location, api, notify, send, $q, authoringWorkspace: AuthoringWorkspaceService, lock) { - var SPIKE_RESOURCE = 'archive_spike', - UNSPIKE_RESOURCE = 'archive_unspike'; - - /** - * Spike given item. - * - * @param {Object} item - * @returns {Promise} - */ - this.spike = function(item) { - return api.update(SPIKE_RESOURCE, item, {state: 'spiked'}) - .then(() => { - if ($location.search()._id === item._id) { - $location.search('_id', null); - } - closeAuthoring(item); - return item; - }, (response) => { - item.error = response; - if (angular.isDefined(response.data._issues) && - angular.isDefined(response.data._issues['validator exception'])) { - notify.error(gettext(response.data._issues['validator exception'])); - } - }); - }; - - /** - * Spike given items. - * - * @param {Object} items - * @returns {Promise} - */ - this.spikeMultiple = function spikeMultiple(items) { - return $q.all(items.map(this.spike)); - }; - - /** - * Unspike given item. - * - * @param {Object} item - */ - this.unspike = function(item) { - return getUnspikeDestination().then((config) => unspike(item, config)); - }; - - function getUnspikeDestination() { - return send.startConfig(); - } - - function unspike(item, config) { - var data = { - task: { - desk: config.desk || null, - stage: config.stage || null, - }, - }; - - return api.update(UNSPIKE_RESOURCE, item, data) - .then(() => { - if ($location.search()._id === item._id) { - $location.search('_id', null); - } - closeAuthoring(item); - return item; - }, (response) => { - item.error = response; - }); - } - - /** - * Unspike given items. - * - * @param {Object} items - */ - this.unspikeMultiple = function unspikeMultiple(items) { - getUnspikeDestination().then((config) => { - items.forEach((item) => { - unspike(item, config); - }); - }); - }; - - /** - * @ngdoc method - * @name spike#closeAuthoring - * @private - * @description Checks if the item is locked (and open by authoring). If yes, unlocks (and closes) it - * @param {Object} item - */ - function closeAuthoring(item) { - const authoringItem = authoringWorkspace.getItem(); - let closeWorkSpace; - - if (authoringItem && authoringItem._id === item._id) { - closeWorkSpace = true; - } - - if (item.state === 'spiked' && closeWorkSpace) { // Unspiked item opened in authoring - authoringWorkspace.close(); - } else if (item.lock_user) { // lock is held by session user (verified at activity.additionalCondition) - lock.unlock(item); - if (closeWorkSpace) { - authoringWorkspace.close(true); - } - } - } -} diff --git a/scripts/apps/archive/services/index.ts b/scripts/apps/archive/services/index.ts index b35d66695f..39bbef1d80 100644 --- a/scripts/apps/archive/services/index.ts +++ b/scripts/apps/archive/services/index.ts @@ -1,5 +1,4 @@ export {DragItemService} from './DragItemService'; export {FamilyService} from './FamilyService'; export {MultiService} from './MultiService'; -export {SpikeService} from './SpikeService'; export {ArchiveService} from './ArchiveService'; diff --git a/scripts/apps/archive/tests/archive.spec.ts b/scripts/apps/archive/tests/archive.spec.ts index a9eadbab42..bac8766faa 100644 --- a/scripts/apps/archive/tests/archive.spec.ts +++ b/scripts/apps/archive/tests/archive.spec.ts @@ -33,95 +33,6 @@ describe('content', () => { Object.assign(appConfig, testConfig); }); - it('can spike items', inject((spike, api, $q) => { - spyOn(api, 'update').and.returnValue($q.when()); - spike.spike(item); - expect(api.update).toHaveBeenCalledWith('archive_spike', item, {state: 'spiked'}); - })); - - it('can unspike items', inject((spike, api, send, $q, $rootScope) => { - var config = {desk: 'foo', stage: 'working'}; - - spyOn(api, 'update').and.returnValue($q.when()); - spyOn(send, 'startConfig').and.returnValue($q.when(config)); - spike.unspike(item); - $rootScope.$digest(); - expect(api.update).toHaveBeenCalledWith('archive_unspike', item, {task: config}); - })); - - it('onSpike middleware is called', - (done) => inject(( - superdesk, - activityService, - privileges, - modal, - lock, - session, - authoringWorkspace: AuthoringWorkspaceService, - config, - metadata, - preferencesService, - ) => { - const extensionDelay = 200; - - const articleEntities = { - onSpike: () => { - return new Promise((resolve) => { - setTimeout(() => { - resolve({}); - }, extensionDelay); - }); - }, - }; - - spyOn(articleEntities, 'onSpike').and.callThrough(); - spyOn(modal, 'createCustomModal').and.callThrough(); // called after middlewares - - registerExtensions( - [ - { - id: 'test-extension', - load: () => Promise.resolve({default: { - activate: () => { - return Promise.resolve({ - contributions: { - entities: { - article: articleEntities, - }, - }, - }); - }, - }}), - }, - ], - superdesk, - modal, - privileges, - lock, - session, - authoringWorkspace, - config, - metadata, - {item: () => false}, - preferencesService, - ).then(() => { - activityService.start(superdesk.activities.spike, {data: {item: {_id: '0'}}}); - - setTimeout(() => { - expect(articleEntities.onSpike).toHaveBeenCalled(); - }); - - setTimeout(() => { - expect(modal.createCustomModal).not.toHaveBeenCalled(); - }, extensionDelay - 50); - - setTimeout(() => { - expect(modal.createCustomModal).toHaveBeenCalled(); - done(); - }, extensionDelay + 50); - }).catch(done.fail); - })); - describe('archive service', () => { beforeEach(inject((desks, session, preferencesService) => { session.identity = {_id: 'user:1'}; @@ -256,28 +167,6 @@ describe('content', () => { })); }); - describe('item preview container', () => { - it('can handle preview:item intent', inject(($rootScope, $compile, superdesk) => { - var scope = $rootScope.$new(); - var elem = $compile('
')(scope); - - scope.$digest(); - - var iscope = elem.isolateScope(); - - expect(iscope.item).toBe(null); - - scope.$apply(() => { - superdesk.intent('preview', 'item', item); - }); - - expect(iscope.item).toBe(item); - - iscope.close(); - expect(iscope.item).toBe(null); - })); - }); - describe('item preview header', () => { it('on toggle sets the header state in local storage', inject(($rootScope, $compile, storage) => { storage.clear(); diff --git a/scripts/apps/archive/utils.ts b/scripts/apps/archive/utils.ts index c11dad12b5..58544da26a 100644 --- a/scripts/apps/archive/utils.ts +++ b/scripts/apps/archive/utils.ts @@ -1,3 +1,7 @@ +/** + * TODO: delete this file and update usages to use sdApi instead + */ + import {IArticle} from 'superdesk-api'; import {PUBLISHED_STATES, KILLED_STATES, ITEM_STATE} from './constants'; diff --git a/scripts/apps/archive/views/list.html b/scripts/apps/archive/views/list.html index 35cda21f45..1db7c0e413 100644 --- a/scripts/apps/archive/views/list.html +++ b/scripts/apps/archive/views/list.html @@ -4,9 +4,8 @@ diff --git a/scripts/apps/archive/views/media-view.html b/scripts/apps/archive/views/media-view.html deleted file mode 100644 index 5d8853c469..0000000000 --- a/scripts/apps/archive/views/media-view.html +++ /dev/null @@ -1,63 +0,0 @@ - diff --git a/scripts/apps/archive/views/preview.html b/scripts/apps/archive/views/preview.html index 1c40b21fd6..c54cbcca47 100644 --- a/scripts/apps/archive/views/preview.html +++ b/scripts/apps/archive/views/preview.html @@ -168,7 +168,7 @@
- +
  • {{item.url}}
    @@ -183,7 +183,7 @@ order="{{editor[field._id].order}}" class="body preview-editor">
    - +
    diff --git a/scripts/apps/archive/views/single-item-preview.html b/scripts/apps/archive/views/single-item-preview.html deleted file mode 100644 index 77d246aec9..0000000000 --- a/scripts/apps/archive/views/single-item-preview.html +++ /dev/null @@ -1,30 +0,0 @@ -
    - - -
    - -
    - - -
    -
    -
    - - - - - -
    -
    -
    - - -
    diff --git a/scripts/apps/authoring-bridge/authoring-api-common.ts b/scripts/apps/authoring-bridge/authoring-api-common.ts new file mode 100644 index 0000000000..b55bf2a1bb --- /dev/null +++ b/scripts/apps/authoring-bridge/authoring-api-common.ts @@ -0,0 +1,110 @@ +import {ITEM_STATE} from 'apps/archive/constants'; +import {runAfterUpdateEvent, runBeforeUpdateMiddlware} from 'apps/authoring/authoring/services/authoring-helpers'; +import {isArticleLockedInCurrentSession} from 'core/get-superdesk-api-implementation'; +import {assertNever} from 'core/helpers/typescript-helpers'; +import ng from 'core/services/ng'; +import {IUnsavedChangesActionWithSaving, showUnsavedChangesPrompt} from 'core/ui/components/prompt-for-unsaved-changes'; +import {IArticle} from 'superdesk-api'; +import {appConfig} from 'appConfig'; + +export interface IAuthoringApiCommon { + saveBefore(current: IArticle, original: IArticle): Promise; + saveAfter(current: IArticle, original: IArticle): void; + closeAuthoring( + original: IArticle, + hasUnsavedChanges: boolean, + save: () => Promise, + unlock: () => Promise, + cancelAutoSave: () => Promise, + doClose: () => void, + ): Promise; + + /** + * Is only meant to be used when there are no unsaved changes + * and item is not locked. + */ + closeAuthoringForce(): void; + + /** + * We need to keep the steps separate because it needs to be called + * separately in Angular. When we remove Angular the closeAuthoring + * and closeAuthoringStep2 will be merged together. + */ + closeAuthoringStep2(scope: any, rootScope: any): Promise; + checkShortcutButtonAvailability: (item: IArticle, dirty?: boolean, personal?: boolean) => boolean; +} + +/** + * Immutable API that is used in both - angularjs and reactjs based authoring code. + */ +export const authoringApiCommon: IAuthoringApiCommon = { + checkShortcutButtonAvailability: (item: IArticle, dirty?: boolean, personal?: boolean): boolean => { + if (personal) { + return appConfig?.features?.publishFromPersonal && item.state !== 'draft'; + } + + return item.task && item.task.desk && item.state !== 'draft' || dirty; + }, + saveBefore: (current, original) => { + return runBeforeUpdateMiddlware(current, original); + }, + saveAfter: (current, original) => { + runAfterUpdateEvent(original, current); + }, + closeAuthoringStep2: (scope: any, rootScope: any): Promise => { + return ng.get('authoring').close( + scope.item, + scope.origItem, + scope.save_enabled(), + () => { + ng.get('authoringWorkspace').close(true); + const itemId = scope.origItem._id; + const storedItemId = localStorage.getItem(`open-item-after-related-closed--${itemId}`); + + rootScope.$broadcast('item:close', itemId); + + /** + * If related item was just created and saved, open the original item + * that triggered the creation of this related item. + */ + if (storedItemId != null) { + return ng.get('autosave').get({_id: storedItemId}).then((resulted) => { + ng.get('authoringWorkspace').open(resulted); + localStorage.removeItem(`open-item-after-related-closed--${itemId}`); + }); + } + }, + ); + }, + closeAuthoring: (original: IArticle, hasUnsavedChanges, save, unlock, cancelAutoSave, doClose) => { + if (!isArticleLockedInCurrentSession(original)) { + return Promise.resolve().then(() => doClose()); + } + + if (hasUnsavedChanges && (original.state !== ITEM_STATE.PUBLISHED && original.state !== ITEM_STATE.CORRECTED)) { + return showUnsavedChangesPrompt(hasUnsavedChanges).then(({action, closePromptFn}) => { + const unlockAndClose = () => unlock().then(() => { + closePromptFn(); + doClose(); + }); + + if (action === IUnsavedChangesActionWithSaving.cancelAction) { + return closePromptFn(); + } else if (action === IUnsavedChangesActionWithSaving.discardChanges) { + return cancelAutoSave().then(() => unlockAndClose()); + } else if (action === IUnsavedChangesActionWithSaving.save) { + return save().then(() => unlockAndClose()); + } else { + assertNever(action); + } + }); + } else { + return unlock().then(() => doClose()); + } + }, + closeAuthoringForce: () => { + ng.get('superdeskFlags').flags.hideMonitoring = false; + + ng.get('authoringWorkspace').close(); + }, +}; diff --git a/scripts/apps/authoring-bridge/receive-patches.ts b/scripts/apps/authoring-bridge/receive-patches.ts new file mode 100644 index 0000000000..73e6c3f066 --- /dev/null +++ b/scripts/apps/authoring-bridge/receive-patches.ts @@ -0,0 +1,33 @@ +import {registerInternalExtension, unregisterInternalExtension} from 'core/helpers/register-internal-extension'; +import {IArticle} from 'superdesk-api'; + +const receivingPatchesInternalExtension = 'receiving-patches-internal-extension'; + +export function registerToReceivePatches(articleId: IArticle['_id'], applyPatch: (patch: Partial) => void) { + registerInternalExtension(receivingPatchesInternalExtension, { + contributions: { + entities: { + article: { + onPatchBefore: (id, patch, dangerousOptions) => { + if ( + articleId === id + && dangerousOptions?.patchDirectlyAndOverwriteAuthoringValues !== true + ) { + applyPatch(patch); + console.info('Article is locked and can\'t be updated via HTTP directly.' + + 'The updates will be added to existing diff in article-edit view instead.'); + + return Promise.reject(); + } else { + return Promise.resolve(patch); + } + }, + }, + }, + }, + }); +} + +export function unregisterFromReceivingPatches() { + unregisterInternalExtension(receivingPatchesInternalExtension); +} diff --git a/scripts/apps/authoring-react/README.md b/scripts/apps/authoring-react/README.md new file mode 100644 index 0000000000..ba1a65e142 --- /dev/null +++ b/scripts/apps/authoring-react/README.md @@ -0,0 +1,35 @@ +# Authoring react + +## Field type + +Field type is an object that implements `ICustomFieldType`. There can be many text fields in an article, like headline, abstract, custom-text-1 etc. but they would all use the same code that a field type provides. + +## Important principles + +There should be no special behavior depending on field ID. It should always be possible to create one more field of the same type and get the same behavior. + +### Field data formats - **storage** and **operational** + +Storage format refers to field data that is stored in `IArticle` when it's received from the API. Operational format is the one being used in runtime. In most cases it will be the same. + +For example if we had a plain text input, we would use a string for storage, and also a string as operational format, since that's what `` uses. + +A different operational format is required, when working with an editor that uses a more complex format that requires custom code to serialize it for storage. For example, draft-js uses `EditorState` as an operational format and requires running additional code in order to serialize it to storage format - `RawDraftContentState`. + +## Storage location of field data + +The data of all fields is stored in `IArticle['extra']` by default. Custom storage location may be specified in field type(`ICustomFieldType`) or field adapter. If a function in field adapter is defined, the one in field type will be ignored. + +## Field adapters + +Field adapters are needed for two purposes. + +1. To allow setting up built-in fields, that are not present in the database as custom fields. +2. To allow storing field data in other location that `IArticle['extra']`. + +The code is simplified by using adapters, since there is only one place where storage details are defined, and the rest of the authoring code doesn't know about it. + + +## Article adapter + +React based authoring isn't using "extra>" prefix for custom fields in `IArticle['fields_meta']`, because there can't be multiple fields with the same ID. I didn't know this when originally implementing `IArticle['fields_meta']`. An upgrade script may be written and prefix dropped completely when angular based authoring code is removed. In the meanwhile, the adapter makes it so the prefix isn't needed in authoring-react code, but is outputted when saving, to make the output compatible with angular based authoring. diff --git a/scripts/apps/authoring-react/article-adapter.ts b/scripts/apps/authoring-react/article-adapter.ts new file mode 100644 index 0000000000..ef77e716cc --- /dev/null +++ b/scripts/apps/authoring-react/article-adapter.ts @@ -0,0 +1,77 @@ +import {IArticle} from 'superdesk-api'; +import {getCustomFieldVocabularies} from 'core/helpers/business-logic'; +import {IOldCustomFieldId} from './interfaces'; + +interface IAuthoringReactArticleAdapter { + /** + * Remove changes done for authoring-react + */ + fromAuthoringReact>(article: T): T; + + /** + * Apply changes required for for authoring-react + */ + toAuthoringReact>(article: T): T; +} + +/** + * There are slight changes in data structure that AuthoringV2 uses. + * + * 1. Fields are generic in AuthoringV2. Field IDs are not used in business logic. + * Adapter moves some custom fields to {@link IArticle} root. + * + * 2. Angular based authoring adds prefixes fields in {@link IArticle.fields_meta} (only {@link IOldCustomFieldId}) + * to prevent possible conflicts of field IDs. It seems though, that validation is in place + * to prevent duplicate IDs, thus prefixing was never necessary. Adapter removes the prefixes. + */ +export function getArticleAdapter(): IAuthoringReactArticleAdapter { + const customFieldVocabularies = getCustomFieldVocabularies(); + + const oldFormatCustomFieldIds: Set = new Set( + customFieldVocabularies + .filter((vocabulary) => vocabulary.field_type === 'text') + .map((vocabulary) => vocabulary._id), + ); + + const adapter: IAuthoringReactArticleAdapter = { + fromAuthoringReact: (_article) => { + // making a copy in order to do immutable updates + let article = {..._article}; + + // Add prefixes + for (const fieldId of Array.from(oldFormatCustomFieldIds)) { + const withPrefix = `extra>${fieldId}`; + + if (article.fields_meta?.hasOwnProperty(fieldId)) { + article.fields_meta[withPrefix] = article.fields_meta[fieldId]; + + delete article.fields_meta[fieldId]; + } + } + + return article; + }, + toAuthoringReact: (_article) => { + let article = {..._article}; // ensure immutability + + if (_article.fields_meta != null) { // ensure immutability + article.fields_meta = {..._article.fields_meta}; + } + + // remove prefixes + for (const fieldId of Array.from(oldFormatCustomFieldIds)) { + const withPrefix = `extra>${fieldId}`; + + if (article.fields_meta?.hasOwnProperty(withPrefix)) { + article.fields_meta[fieldId] = article.fields_meta[withPrefix]; + + delete article.fields_meta[withPrefix]; + } + } + + return article; + }, + }; + + return adapter; +} diff --git a/scripts/apps/authoring-react/article-widgets/README.MD b/scripts/apps/authoring-react/article-widgets/README.MD new file mode 100644 index 0000000000..88b21495e5 --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/README.MD @@ -0,0 +1,17 @@ +# About + +These widgets are react-only widgets that will appear in both - current and new versions of authoring when enabled. +The goal is to re-implement existing angular widgets in react. We can run react code inside angularjs, but not vice-versa. +When a widget is re-implemented in react, we can delete the angular code +and enable the widget in `../manage-widget-registration.ts`. + +# Adding a widget + +* Add a file or folder per widget. +* Export an object or a function returning the object implementing `IAuthoringSideWidget` interface. +* Import it in `../manage-widget-registration.ts` +* Use `AuthoringWidgetLayout` as a root widget component. + +To update the article that is being edited, call `sdApi.article.patch`. + +An example is available in ./demo-widget.tsx diff --git a/scripts/apps/authoring-react/article-widgets/comments/index.tsx b/scripts/apps/authoring-react/article-widgets/comments/index.tsx new file mode 100644 index 0000000000..396be6b2af --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/comments/index.tsx @@ -0,0 +1,73 @@ +/* eslint-disable react/no-multi-comp */ +import React from 'react'; +import {IArticleSideWidget, IComment, IExtensionActivationResult, IRestApiResponse} from 'superdesk-api'; +import {gettext} from 'core/utils'; +import CommentsWidget from '../../generic-widgets/comments/CommentsWidget'; +import {httpRequestJsonLocal} from 'core/helpers/network'; +// Can't call `gettext` in the top level +const getLabel = () => gettext('Comments'); + +type IProps = React.ComponentProps< + IExtensionActivationResult['contributions']['authoringSideWidgets'][0]['component'] +>; + +class Component extends React.PureComponent { + render() { + return ( + { + const itemId = this.props.article?._id; + + if (itemId == null) { + return Promise.resolve([]); + } + + const criteria = { + where: { + item: itemId, + }, + embedded: {user: 1}, + }; + + return httpRequestJsonLocal>({ + method: 'GET', + path: '/item_comments', + urlParams: criteria, + }).then(({_items}) => _items); + }} + addComment={(text) => { + return httpRequestJsonLocal({ + method: 'POST', + path: '/item_comments', + payload: { + item: this.props.article._id, + text: text, + }, + }); + }} + /> + ); + } +} + +export function getCommentsWidget() { + const widget: IArticleSideWidget = { + _id: 'comments-widget', + label: getLabel(), + order: 3, + icon: 'chat', + component: Component, + isAllowed: (item) => item._type !== 'legal_archive', + }; + + return widget; +} diff --git a/scripts/apps/authoring-react/article-widgets/demo-widget.tsx b/scripts/apps/authoring-react/article-widgets/demo-widget.tsx new file mode 100644 index 0000000000..6ed1ce9ed3 --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/demo-widget.tsx @@ -0,0 +1,58 @@ +import React from 'react'; +import {IArticleSideWidget, IArticle, IExtensionActivationResult} from 'superdesk-api'; +import {Button} from 'superdesk-ui-framework'; +import {sdApi} from 'api'; +import {gettext} from 'core/utils'; +import {AuthoringWidgetHeading} from 'apps/dashboard/widget-heading'; +import {AuthoringWidgetLayout} from 'apps/dashboard/widget-layout'; + +// Can't call `gettext` in the top level +const getLabel = () => gettext('Demo widget'); + +type IProps = React.ComponentProps< + IExtensionActivationResult['contributions']['authoringSideWidgets'][0]['component'] +>; + +class DemoWidget extends React.PureComponent { + render() { + return ( + + )} + body={( +
    +
    + )} + footer={( +
    test footer
    + )} + /> + ); + } +} + +export function getDemoWidget() { + const metadataWidget: IArticleSideWidget = { + _id: 'demo-widget', + label: getLabel(), + order: 2, + icon: 'info', + component: DemoWidget, + }; + + return metadataWidget; +} diff --git a/scripts/apps/authoring-react/article-widgets/find-and-replace.tsx b/scripts/apps/authoring-react/article-widgets/find-and-replace.tsx new file mode 100644 index 0000000000..f744c6e3dc --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/find-and-replace.tsx @@ -0,0 +1,176 @@ +import React from 'react'; +import {IArticleSideWidget, IArticle, IExtensionActivationResult} from 'superdesk-api'; +import {gettext} from 'core/utils'; +import {AuthoringWidgetHeading} from 'apps/dashboard/widget-heading'; +import {AuthoringWidgetLayout} from 'apps/dashboard/widget-layout'; +import {Input, Button, IconButton, Switch} from 'superdesk-ui-framework/react'; +import {dispatchEditorEvent} from '../authoring-react-editor-events'; +import {Spacer} from 'core/ui/components/Spacer'; +import {throttle} from 'lodash'; + +// Can't call `gettext` in the top level +const getLabel = () => gettext('Find and Replace'); + +type IProps = React.ComponentProps< + IExtensionActivationResult['contributions']['authoringSideWidgets'][0]['component'] +>; + +interface IState { + findValue: string; + replaceValue: string; + caseSensitive: boolean; +} + +/** + * Current implementation of find-replace only supports one field. + */ +export const editorId = 'body_html'; + +class FindAndReplaceWidget extends React.PureComponent { + private scheduleHighlightingOfMatches: () => void; + + constructor(props: IProps) { + super(props); + + this.state = { + findValue: '', + replaceValue: '', + caseSensitive: false, + }; + + this.highlightMatches.bind(this); + + this.scheduleHighlightingOfMatches = throttle( + this.highlightMatches, + 500, + {leading: false}, + ); + } + + private highlightMatches() { + dispatchEditorEvent('find_and_replace__find', { + editorId, + text: this.state.findValue, + caseSensitive: this.state.caseSensitive, + }); + } + + componentWillUnmount() { + // remove highlights from editor + dispatchEditorEvent('find_and_replace__find', { + editorId, + text: '', + caseSensitive: false, + }); + } + + render() { + return ( + + )} + body={( + + { + this.setState({findValue}); + this.scheduleHighlightingOfMatches(); + }} + /> + + { + this.setState({replaceValue}); + }} + /> + + { + this.setState({caseSensitive}); + this.scheduleHighlightingOfMatches(); + }} + /> + +
    + + { + dispatchEditorEvent('find_and_replace__find_prev', {editorId}); + }} + icon="chevron-left-thin" + /> + + { + dispatchEditorEvent('find_and_replace__find_next', {editorId}); + }} + icon="chevron-right-thin" + /> + + + +
    +
    + )} + /> + ); + } +} + +export function getFindAndReplaceWidget() { + const metadataWidget: IArticleSideWidget = { + _id: 'find-and-replace-widget', + label: getLabel(), + order: 1, + icon: 'find-replace', + component: FindAndReplaceWidget, + }; + + return metadataWidget; +} diff --git a/scripts/apps/authoring-react/article-widgets/inline-comments.tsx b/scripts/apps/authoring-react/article-widgets/inline-comments.tsx new file mode 100644 index 0000000000..3f5ffbea6c --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/inline-comments.tsx @@ -0,0 +1,44 @@ +/* eslint-disable react/no-multi-comp */ + +import React from 'react'; +import {IArticleSideWidget, IExtensionActivationResult, IArticle} from 'superdesk-api'; +import {gettext} from 'core/utils'; +import {InlineCommentsWidget} from '../generic-widgets/inline-comments'; + +// Can't call `gettext` in the top level +const getLabel = () => gettext('Inline comments'); + +type IProps = React.ComponentProps< + IExtensionActivationResult['contributions']['authoringSideWidgets'][0]['component'] +>; + +class InlineCommentsWidgetWrapper extends React.PureComponent { + render() { + return ( + + entityId={this.props.article._id} + readOnly={this.props.readOnly} + contentProfile={this.props.contentProfile} + fieldsData={this.props.fieldsData} + authoringStorage={this.props.authoringStorage} + fieldsAdapter={this.props.fieldsAdapter} + storageAdapter={this.props.storageAdapter} + onFieldsDataChange={this.props.onFieldsDataChange} + handleUnsavedChanges={this.props.handleUnsavedChanges} + /> + ); + } +} + +export function getInlineCommentsWidget() { + const metadataWidget: IArticleSideWidget = { + _id: 'inline-comments-widget', + label: getLabel(), + order: 2, + icon: 'comments', + component: InlineCommentsWidgetWrapper, + isAllowed: (item) => item._type !== 'legal_archive', + }; + + return metadataWidget; +} diff --git a/scripts/apps/authoring-react/article-widgets/metadata/AnnotationsPreview.tsx b/scripts/apps/authoring-react/article-widgets/metadata/AnnotationsPreview.tsx new file mode 100644 index 0000000000..12de91baed --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/metadata/AnnotationsPreview.tsx @@ -0,0 +1,43 @@ +import {getAllAnnotations} from 'apps/archive/directives/HtmlPreview'; +import {Spacer} from 'core/ui/components/Spacer'; +import {gettext} from 'core/utils'; +import React from 'react'; +import {IArticle} from 'superdesk-api'; +import {Label, ToggleBox} from 'superdesk-ui-framework/react'; +import './annotations-preview.scss'; + +interface IProps { + article: IArticle; +} + +export class AnnotationsPreview extends React.Component { + render(): React.ReactNode { + const {article} = this.props; + + return ( +
    +
    + + { + (article.annotations?.length ?? 0) > 0 && ( + getAllAnnotations(article).map((annotation) => ( + + + )) + ) + } + +
    + ); + } +} diff --git a/scripts/apps/authoring-react/article-widgets/metadata/annotations-preview.scss b/scripts/apps/authoring-react/article-widgets/metadata/annotations-preview.scss new file mode 100644 index 0000000000..04b7c31947 --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/metadata/annotations-preview.scss @@ -0,0 +1,6 @@ +.annotation-body-react { + border-bottom: 1px dotted; + p { + display: inline; + } +} diff --git a/scripts/apps/authoring-react/article-widgets/metadata/metadata-item.tsx b/scripts/apps/authoring-react/article-widgets/metadata/metadata-item.tsx new file mode 100644 index 0000000000..1dc1612c09 --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/metadata/metadata-item.tsx @@ -0,0 +1,26 @@ +import {Spacer} from 'core/ui/components/Spacer'; +import React from 'react'; +import {ContentDivider, Heading} from 'superdesk-ui-framework/react'; + +interface IProps { + label: string; + value: string | number | JSX.Element; +} + +export class MetadataItem extends React.Component { + render(): React.ReactNode { + const {label, value} = this.props; + + return ( + <> + + + {label.toUpperCase()} + +
    {value}
    +
    + + + ); + } +} diff --git a/scripts/apps/authoring-react/article-widgets/metadata/metadata.tsx b/scripts/apps/authoring-react/article-widgets/metadata/metadata.tsx new file mode 100644 index 0000000000..0ab69a3f86 --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/metadata/metadata.tsx @@ -0,0 +1,476 @@ +import React, {Fragment} from 'react'; +import {IArticleSideWidget, IExtensionActivationResult, IVocabularyItem} from 'superdesk-api'; +import {gettext} from 'core/utils'; +import {AuthoringWidgetHeading} from 'apps/dashboard/widget-heading'; +import {AuthoringWidgetLayout} from 'apps/dashboard/widget-layout'; +import {Spacer} from 'core/ui/components/Spacer'; +import {Input, Select, Switch, Option, Heading, ContentDivider, Label} from 'superdesk-ui-framework/react'; +import {MetadataItem} from './metadata-item'; +import {dataApi} from 'core/helpers/CrudManager'; +import {ILanguage} from 'superdesk-interfaces/Language'; +import {DateTime} from 'core/ui/components/DateTime'; +import {vocabularies} from 'api/vocabularies'; +import Datetime from 'core/datetime/datetime'; +import {sdApi} from 'api'; +import {StateComponent} from 'apps/search/components/fields/state'; +import {AnnotationsPreview} from './AnnotationsPreview'; + +// Can't call `gettext` in the top level +const getLabel = () => gettext('Metadata'); + +type IProps = React.ComponentProps< + IExtensionActivationResult['contributions']['authoringSideWidgets'][0]['component'] +>; + +interface IState { + languages: Array; +} + +class MetadataWidget extends React.PureComponent { + constructor(props: IProps) { + super(props); + + this.state = { + languages: [], + }; + } + + componentDidMount(): void { + dataApi.query( + 'languages', + 1, + {field: 'language', direction: 'ascending'}, + {}, + ).then(({_items}) => { + this.setState({ + languages: _items, + }); + }); + } + + render() { + const {article} = this.props; + + const { + flags, + usageterms, + pubstatus, + expiry, + urgency, + priority, + word_count, + source, + anpa_take_key, + genre, + dateline, + slugline, + byline, + sign_off, + guid, + unique_name, + type, + language, + copyrightholder, + copyrightnotice, + creditline, + original_source, + ingest_provider_sequence, + ingest_provider, + keywords, + signal, + anpa_category, + place, + ednote, + _current_version, + firstcreated, + versioncreated, + renditions, + original_id, + originalCreator, + versioncreator, + rewritten_by, + } = article; + + const {onArticleChange} = this.props; + + const allVocabularies = sdApi.vocabularies.getAll(); + + return ( + + )} + body={( + + + + {gettext('Not For Publication')} + + { + onArticleChange({ + ...article, + flags: { + ...flags, + marked_for_not_publication: !flags.marked_for_not_publication, + }, + }); + }} + value={flags.marked_for_not_publication} + /> + + + + + + + {gettext('Legal')} + + { + onArticleChange({ + ...article, + flags: {...flags, marked_for_legal: !flags.marked_for_legal}, + }); + }} + value={flags.marked_for_legal} + /> + + + + + { + onArticleChange({ + ...article, + usageterms: value, + }); + }} + /> + + + + + + + + {(pubstatus?.length ?? 0) > 0 && ( + + )} + + {(original_source?.length ?? 0) > 0 && ( + + )} + + {(copyrightholder?.length ?? 0) > 0 && ( + + )} + + {(copyrightnotice?.length ?? 0) > 0 && ( + + )} + + {(creditline?.length ?? 0) > 0 && ( + + )} + + { + <> + + + {gettext('State').toUpperCase()} + + + + {article.embargo && ( + + + + + } + + {ingest_provider != null && ( + + )} + + { + (ingest_provider_sequence?.length ?? 0) > 0 && ( + + ) + } + + {expiry && ( + } + /> + )} + + {(slugline?.length ?? 0) > 0 && } + + {(urgency?.length ?? 0) > 0 && } + + {priority && } + + {word_count > 0 && } + + {keywords && ( + + )} + + {(source?.length ?? 0) > 0 && } + + + + { + signal && ( + + {(signal.map(({name, qcode}) => ( + {name ?? qcode} + )))} +
    + )} + /> + ) + } + + { + anpa_category?.name != null && ( + + ) + } + + { + allVocabularies + .filter((cv) => article[cv.schema_field] != null) + .toArray() + .map((vocabulary) => ( + + )) + } + + { + (genre.length ?? 0) > 0 + && allVocabularies.map((v) => v.schema_field).includes('genre') === false + && ( + + ) + } + + { + (place.length ?? 0) > 0 + && allVocabularies.map((v) => v.schema_field).includes('place') === false + && ( + + ) + } + + {(ednote?.length ?? 0) > 0 && } + + + + {gettext('Dateline').toUpperCase()} + + + / + {dateline?.located.city} + + + + + + + + + + {_current_version && } + + {firstcreated && ( + + )} + /> + )} + + {versioncreated && ( + } + /> + )} + + + + {(originalCreator?.length ?? 0) > 0 && ( + + )} + + {(versioncreator?.length ?? 0) > 0 && ( + + )} + + + + { + onArticleChange({ + ...article, + unique_name: value, + }); + }} + /> + + + + + + { + renditions?.original != null && ( + + ) + } + + { + article.type === 'picture' + && article.archive_description !== article.description_text + && ( + + ) + } + + )} + /> + ); + } +} + +export function getMetadataWidget() { + const metadataWidget: IArticleSideWidget = { + _id: 'metadata-widget', + label: getLabel(), + order: 1, + icon: 'info', + component: MetadataWidget, + }; + + return metadataWidget; +} diff --git a/scripts/apps/authoring-react/article-widgets/suggestions.tsx b/scripts/apps/authoring-react/article-widgets/suggestions.tsx new file mode 100644 index 0000000000..355e20e955 --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/suggestions.tsx @@ -0,0 +1,242 @@ +/* eslint-disable react/no-multi-comp */ + +import React from 'react'; +import {IArticleSideWidget, IExtensionActivationResult, IUser, IEditor3ValueOperational} from 'superdesk-api'; +import {gettext} from 'core/utils'; +import {AuthoringWidgetHeading} from 'apps/dashboard/widget-heading'; +import {AuthoringWidgetLayout} from 'apps/dashboard/widget-layout'; +import {EmptyState, Label} from 'superdesk-ui-framework/react'; +import {getCustomEditor3Data} from 'core/editor3/helpers/editor3CustomData'; +import {store} from 'core/data'; +import {Card} from 'core/ui/components/Card'; +import {UserAvatar} from 'apps/users/components/UserAvatar'; +import {TimeElem} from 'apps/search/components'; +import {Spacer, SpacerBlock} from 'core/ui/components/Spacer'; +import {getLocalizedTypeText} from 'apps/authoring/track-changes/suggestions'; + +// Can't call `gettext` in the top level +const getLabel = () => gettext('Resolved suggestions'); + +type IProps = React.ComponentProps< + IExtensionActivationResult['contributions']['authoringSideWidgets'][0]['component'] +>; + +interface ISuggestion { + resolutionInfo: { + accepted: boolean; + date: string; + resolverUserId: IUser['_id']; + }; + + suggestionInfo: { + author: IUser['_id']; + date: string; + selection: {}; // serialized SelectionState + styleName: string; + suggestionText: string; + type: string; + blockType?: string; + link?: { // only for link suggestions + href: string; + } + }; + + suggestionText: string; + oldText?: string; // used with replace suggestion +} + +class Suggestion extends React.PureComponent<{suggestion: ISuggestion}> { + render() { + const {suggestionInfo, suggestionText, oldText, resolutionInfo} = this.props.suggestion; + const suggestionAuthor = + store.getState().entities.users[suggestionInfo.author]; + const suggestionResolver = + store.getState().entities.users[resolutionInfo.resolverUserId]; + + return ( + + +
    + + + +
    + {suggestionAuthor.display_name}: + +
    + +
    +
    +
    +
    + +
    + { + resolutionInfo.accepted + ? (
    +
    + + + +
    + { + suggestionInfo.type === 'REPLACE_SUGGESTION' + ? ( + "${oldText}"`, + y: `"${suggestionText}"`, + }, + ), + }} + /> + ) + : ( + + { + getLocalizedTypeText( + suggestionInfo.type, + suggestionInfo.blockType, + ) + } + :  + + "{suggestionText}" + + { + suggestionInfo.type === 'ADD_LINK_SUGGESTION' && ( + +  {suggestionInfo.link.href} + + ) + } + + ) + } +
    + + + +
    + { + resolutionInfo.accepted + ? gettext('Accepted by {{user}}', {user: suggestionResolver.display_name}) + : gettext('Rejected by {{user}}', {user: suggestionResolver.display_name}) + } + +   + + +
    +
    + ); + } +} + +class SuggestionsWidget extends React.PureComponent { + constructor(props: IProps) { + super(props); + + this.getEditor3Fields = this.getEditor3Fields.bind(this); + this.getResolvedSuggestions = this.getResolvedSuggestions.bind(this); + } + + getEditor3Fields() { + const {contentProfile} = this.props; + const allFields = contentProfile.header.merge(contentProfile.content); + + return allFields.filter((field) => field.fieldType === 'editor3').toArray(); + } + + getResolvedSuggestions() { + const {fieldsData} = this.props; + + return this.getEditor3Fields().map((field) => { + const value = fieldsData.get(field.id) as IEditor3ValueOperational; + + return { + fieldId: field.id, + suggestions: (getCustomEditor3Data( + value.contentState, + 'RESOLVED_SUGGESTIONS_HISTORY', + ) ?? []) as Array, + }; + }).filter(({suggestions}) => suggestions.length > 0); + } + + render() { + const {contentProfile} = this.props; + const allFields = contentProfile.header.merge(contentProfile.content); + const resolvedSuggestions = this.getResolvedSuggestions(); + + const widgetBody: JSX.Element = resolvedSuggestions.length > 0 + ? ( +
    + + { + resolvedSuggestions.map(({fieldId, suggestions}, i) => { + return ( +
    +
    + {allFields.get(fieldId).name} +
    + + + + + { + suggestions.map((suggestion, j) => ( + + )) + } + +
    + ); + }) + } +
    +
    + ) + : ( + + ); + + return ( + + )} + body={widgetBody} + background="grey" + /> + ); + } +} + +export function getSuggestionsWidget() { + const metadataWidget: IArticleSideWidget = { + _id: 'editor3-suggestions-widget', + label: getLabel(), + order: 3, + icon: 'suggestion', + component: SuggestionsWidget, + isAllowed: (item) => item._type !== 'legal_archive' && item._type !== 'archived', + }; + + return metadataWidget; +} diff --git a/scripts/apps/authoring-react/article-widgets/translations/TranslationsBody.tsx b/scripts/apps/authoring-react/article-widgets/translations/TranslationsBody.tsx new file mode 100644 index 0000000000..d5dc6c8954 --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/translations/TranslationsBody.tsx @@ -0,0 +1,62 @@ +import React from 'react'; +import ng from 'core/services/ng'; +import {IArticle} from 'superdesk-api'; + +interface IProps { + item: IArticle; + wrapperTemplate: React.ComponentType<{children: Array}>; + translationTemplate: React.ComponentType<{translation: IArticle, getTranslatedFromLanguage: () => string}>; +} + +interface IState { + translations: Array | null; + translationsLookup: Dictionary; +} + +export class TranslationsBody extends React.PureComponent { + componentDidMount() { + const {item} = this.props; + + ng.get('TranslationService').getTranslations(item) + .then((response) => { + const translations: Array = response._items; + + this.setState({ + translations: translations, + translationsLookup: translations.reduce((result, reference) => { + result[reference._id] = reference; + return result; + }, {}), + }); + }); + } + + render(): React.ReactNode { + if (this.state?.translations == null || this.state?.translationsLookup == null) { + return null; + } + + const sortOldestFirst = (a: IArticle, b: IArticle) => + new Date(b.firstcreated) > new Date(a.firstcreated) ? -1 : 1; + const WrapperTemplate = this.props.wrapperTemplate; + const TranslationTemplate = this.props.translationTemplate; + + return ( + + { + this.state.translations.sort(sortOldestFirst).map((translation: IArticle, i) => { + return ( + this.state.translationsLookup[translation.translated_from]?.language + } + /> + ); + }) + } + + ); + } +} diff --git a/scripts/apps/authoring-react/article-widgets/translations/translations.tsx b/scripts/apps/authoring-react/article-widgets/translations/translations.tsx new file mode 100644 index 0000000000..0bab637a02 --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/translations/translations.tsx @@ -0,0 +1,101 @@ +import React from 'react'; +import {RelativeDate} from 'core/datetime/relativeDate'; +import {state as State} from 'apps/search/components/fields/state'; +import {IArticle, IArticleSideWidget, IExtensionActivationResult} from 'superdesk-api'; +import {gettext} from 'core/utils'; +import {openArticle} from 'core/get-superdesk-api-implementation'; +import {AuthoringWidgetLayout} from 'apps/dashboard/widget-layout'; +import {AuthoringWidgetHeading} from 'apps/dashboard/widget-heading'; +import {Card} from 'core/ui/components/Card'; +import {Spacer, SpacerBlock} from 'core/ui/components/Spacer'; +import {Label} from 'superdesk-ui-framework'; +import {TranslationsBody} from './TranslationsBody'; + +const getLabel = () => gettext('Translations'); + +type IProps = React.ComponentProps< + IExtensionActivationResult['contributions']['authoringSideWidgets'][0]['component'] +>; + +class Translations extends React.Component { + render() { + return ( + + )} + body={( + + {children} + } + translationTemplate={({translation, getTranslatedFromLanguage}) => ( + +
    openArticle(translation._id, 'edit')}> +
    + + {translation.language} + + {translation.headline} + + + + + + + +
    + { + translation.translated_from == null + ? ( +
    +
    + +
    +
    +
    +
    +
    + )} + /> + )} + /> + ); + } +} + +export function getTranslationsWidget() { + const metadataWidget: IArticleSideWidget = { + _id: 'translation-widget', + label: getLabel(), + order: 2, + icon: 'web', + component: Translations, + }; + + return metadataWidget; +} diff --git a/scripts/apps/authoring-react/article-widgets/versions-and-item-history/history-tab.tsx b/scripts/apps/authoring-react/article-widgets/versions-and-item-history/history-tab.tsx new file mode 100644 index 0000000000..34202c608f --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/versions-and-item-history/history-tab.tsx @@ -0,0 +1,554 @@ +import React from 'react'; +import {Map} from 'immutable'; +import {IExtensionActivationResult, IRestApiResponse} from 'superdesk-api'; +import {getHistoryItems, IHistoryItem, getOperationLabel} from 'apps/authoring/versioning/history/HistoryController'; +import {TimeElem} from 'apps/search/components'; +import {gettext} from 'core/utils'; +import {Spacer} from 'core/ui/components/Spacer'; +import {Card} from 'core/ui/components/Card'; +import {openArticle} from 'core/get-superdesk-api-implementation'; +import {IHighlight} from 'apps/highlights/services/HighlightsService'; +import {httpRequestJsonLocal} from 'core/helpers/network'; +import {Button, ToggleBox} from 'superdesk-ui-framework/react'; +import {TransmissionDetails} from './transmission-details'; + +type IProps = React.ComponentProps< + IExtensionActivationResult['contributions']['authoringSideWidgets'][0]['component'] +>; + +interface IState { + historyItems: Array | null; + highlights: Map | null; +} + +export class HistoryTab extends React.PureComponent { + constructor(props: IProps) { + super(props); + + this.state = { + historyItems: null, + highlights: null, + }; + + this.getHighlightsLabel = this.getHighlightsLabel.bind(this); + } + + getHighlightsLabel(id: IHighlight['_id'], fallbackLabel: string): string { + return this.state.highlights.get(id)?.name ?? fallbackLabel; + } + + componentDidMount() { + Promise.all([ + getHistoryItems(this.props.article), + httpRequestJsonLocal>({method: 'GET', path: '/highlights'}), + ]).then(([historyItems, highlightsResponse]) => { + this.setState({ + historyItems, + highlights: Map(highlightsResponse._items.map((item) => [item._id, item])), + }); + }); + } + + render() { + const {historyItems} = this.state; + + if (historyItems == null) { + return null; + } + + const BaseHistoryItem: React.ComponentType<{items: Array, current: number}> = (props) => { + const {items, current, children} = props; + const item = items[current]; + const itemPrevious = items[current - 1]; + const showVersion = item.version > 0 && item.version !== itemPrevious?.version; + + return ( + + + +
    + +
    + + { + showVersion && ( +
    + {gettext('Version: {{n}}', {n: item.version})} +
    + ) + } +
    + + {children} +
    +
    + ); + }; + + return ( + + { + historyItems.map((item, i) => { + if (item.operation === 'create') { + return ( + +
    + {gettext('Created by')} +   + {item.displayName} +
    + + { + item.desk != null && ( +
    {item.desk} / {item.stage}
    + ) + } +
    + ); + } else if (item.operation === 'update') { + if (item.update.operation === 'deschedule') { + return ( + +
    + {gettext('Descheduled by')} +   + {item.displayName} +
    + +
    + {gettext('Updated fields:')} +   + {item.fieldsUpdated} +
    +
    + ); + } else { + return ( + +
    + {gettext('Updated by')} +   + {item.displayName} +
    + +
    + {gettext('Updated fields:')} +   + {item.fieldsUpdated} +
    +
    + ); + } + } else if (item.operation === 'duplicated_from') { + return ( + +
    + {gettext('Duplicated by')} +   + {item.displayName} +
    + + { + item.update.duplicate_id != null && ( +
    +
    + ) + } +
    + ); + } else if (item.operation === 'duplicate') { + return ( + +
    + {gettext('Duplicate created by')} +   + {item.displayName} +
    + + { + item.update.duplicate_id != null && ( +
    +
    + ) + } +
    + ); + } else if (item.operation === 'translate') { + return ( + +
    + {gettext('Translated by')} +   + {item.displayName} +
    + + { + item.update.duplicate_id != null && ( +
    +
    + ) + } +
    + ); + } else if (item.operation === 'spike') { + return ( + +
    + {gettext('Spiked by')} +   + {item.displayName} +
    + + { + item.desk != null && ( +
    {gettext('from:')} {item.desk} / {item.stage}
    + ) + } +
    + ); + } else if (item.operation === 'unspike') { + return ( + +
    + {gettext('Unspiked by')} +   + {item.displayName} +
    + + { + item.desk != null && ( +
    {gettext('from:')} {item.desk} / {item.stage}
    + ) + } +
    + ); + } else if (item.operation === 'move') { + return ( + +
    + {gettext('Moved by')} +   + {item.displayName} +
    + + { + item.desk != null && ( +
    {gettext('from:')} {item.desk} / {item.stage}
    + ) + } +
    + ); + } else if (item.operation === 'fetch') { + return ( + +
    + {gettext('Fetched by')} +   + {item.displayName} +
    + + { + item.desk != null && ( +
    {gettext('from:')} {item.desk} / {item.stage}
    + ) + } +
    + ); + } else if (item.operation === 'mark' && item.update.highlight_id !== null) { + return ( + +
    + {gettext( + 'Marked for highlight {{x}} by {{user}}', + { + x: this.getHighlightsLabel( + item.update.highlight_id, + item.update.highlight_name, + ), + user: item.displayName, + }, + )} +
    +
    + ); + } else if (item.operation === 'unmark' && item.update.highlight_id !== null) { + return ( + +
    + {gettext( + 'Unmarked from highlight({{x}}) by {{user}}', + { + x: this.getHighlightsLabel( + item.update.highlight_id, + item.update.highlight_name, + ), + user: item.displayName, + }, + )} +
    +
    + ); + } else if (item.operation === 'mark' && item.update.desk_id !== null) { + return ( + +
    + {gettext( + 'Marked for desk {{x}} by {{user}}', + { + x: this.getHighlightsLabel( + item.update.highlight_id, + item.update.highlight_name, + ), + user: item.displayName, + }, + )} +
    +
    + ); + } else if (item.operation === 'unmark' && item.update.desk_id !== null) { + return ( + +
    + {gettext( + 'Unmarked from desk({{x}}) by {{user}}', + { + x: this.getHighlightsLabel( + item.update.highlight_id, + item.update.highlight_name, + ), + user: item.displayName, + }, + )} +
    +
    + ); + } else if (item.operation === 'export_highlight') { + return ( + +
    + {gettext('Exported by')} +   + {item.displayName} +
    + + { + item.update.highlight_id != null && ( +
    + {gettext( + 'from highlight: {{x}}', + { + x: this.getHighlightsLabel( + item.update.highlight_id, + item.update.highlight_name, + ), + }, + )} +
    + ) + } + +
    + ); + } else if (item.operation === 'create_highlight' && item.update.highlight_id != null) { + return ( + +
    + {gettext( + 'Created from highlight({{x}}) by {{user}}', + { + x: this.getHighlightsLabel( + item.update.highlight_id, + item.update.highlight_name, + ), + user: item.displayName, + }, + )} +
    + +
    + ); + } else if (item.operation === 'link') { + return ( + +
    + {gettext('Linked by {{user}}', {user: item.displayName})} +
    + +
    +
    + +
    + ); + } else if (item.operation === 'take') { + return ( + +
    + {gettext( + 'Take created by {{user}}', + { + user: item.displayName, + }, + )} +
    + + +
    {gettext('Taken as a rewrite of')}
    +
    + ) + } + + ); + } else if ( + item.operation === 'publish' + || item.operation === 'correct' + || item.operation === 'kill' + || item.operation === 'takedown' + || item.operation === 'unpublish' + ) { + return ( + + + {getOperationLabel(item.operation, item.update.state)} {item.displayName} + + + + + + + ); + } else { + return null; + } + }) + } + + ); + } +} diff --git a/scripts/apps/authoring-react/article-widgets/versions-and-item-history/index.tsx b/scripts/apps/authoring-react/article-widgets/versions-and-item-history/index.tsx new file mode 100644 index 0000000000..b6f1398a8d --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/versions-and-item-history/index.tsx @@ -0,0 +1,84 @@ +import React from 'react'; +import { + IArticleSideWidget, + IExtensionActivationResult, +} from 'superdesk-api'; +import {gettext} from 'core/utils'; +import {AuthoringWidgetHeading} from 'apps/dashboard/widget-heading'; +import {AuthoringWidgetLayout} from 'apps/dashboard/widget-layout'; +import {assertNever} from 'core/helpers/typescript-helpers'; +import {TabList} from 'core/ui/components/tabs'; +import {HistoryTab} from './history-tab'; +import {VersionsTab} from './versions-tab'; + +// Can't call `gettext` in the top level +const getLabel = () => gettext('Versions and item history'); + +type IProps = React.ComponentProps< + IExtensionActivationResult['contributions']['authoringSideWidgets'][0]['component'] +>; + +interface IState { + selectedTab: 'versions' | 'history'; +} + +class VersionsAndItemHistoryWidget extends React.PureComponent { + constructor(props: IProps) { + super(props); + + this.state = { + selectedTab: 'versions', + }; + } + + render() { + return ( + { + this.setState({selectedTab}); + }} + selectedTabId={this.state.selectedTab} + /> + )} + /> + )} + body={(() => { + if (this.state.selectedTab === 'history') { + return ( + + ); + } else if (this.state.selectedTab === 'versions') { + return ( + + ); + } else { + assertNever(this.state.selectedTab); + } + })()} + background="grey" + /> + ); + } +} + +export function getVersionsAndItemHistoryWidget() { + const widget: IArticleSideWidget = { + _id: 'versions-and-item-history', + label: getLabel(), + order: 4, + icon: 'history', + component: VersionsAndItemHistoryWidget, + }; + + return widget; +} diff --git a/scripts/apps/authoring-react/article-widgets/versions-and-item-history/transmission-details.tsx b/scripts/apps/authoring-react/article-widgets/versions-and-item-history/transmission-details.tsx new file mode 100644 index 0000000000..95b2f7e341 --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/versions-and-item-history/transmission-details.tsx @@ -0,0 +1,139 @@ +import React from 'react'; +import {IHistoryItem} from 'apps/authoring/versioning/history/HistoryController'; +import {IArticle, IRestApiResponse} from 'superdesk-api'; +import {httpRequestJsonLocal} from 'core/helpers/network'; +import {IPublishQueueItem} from 'superdesk-interfaces/PublishQueueItem'; +import {gettext} from 'core/utils'; +import {IconButton} from 'superdesk-ui-framework/react'; +import {showModal} from '@superdesk/common'; +import {Modal} from 'core/ui/components/Modal/Modal'; +import {ModalHeader} from 'core/ui/components/Modal/ModalHeader'; +import {ModalBody} from 'core/ui/components/Modal/ModalBody'; +import {TimeElem} from 'apps/search/components/TimeElem'; +import {Spacer} from 'core/ui/components/Spacer'; + +interface IProps { + article: IArticle; + historyItem: IHistoryItem; +} + +interface IState { + queueItems: Array | null; +} + +function tryShowingFormattedJson(maybeJson: string) { + try { + const parsed = JSON.parse(maybeJson); + + return ( +
    {JSON.stringify(parsed, null, 2)}
    + ); + } catch { + return {maybeJson}; + } +} + +export class TransmissionDetails extends React.PureComponent { + constructor(props: IProps) { + super(props); + + this.state = { + queueItems: null, + }; + } + + componentDidMount() { + httpRequestJsonLocal>({ + method: 'GET', + path: this.props.article._type === 'legal_archive' ? '/legal_publish_queue' : '/publish_queue', + urlParams: { + max_results: 20, + where: { + $and: [ + {item_id: this.props.historyItem.item_id}, + {item_version: this.props.historyItem.version}, + ], + }, + }, + }).then((res) => { + this.setState({queueItems: res._items}); + }); + } + + render() { + const {queueItems} = this.state; + + if (queueItems == null) { + return null; + } + + if (queueItems.length < 1) { + return ( +
    {gettext('Item has not been transmitted to any subscriber')}
    + ); + } + + return ( + {/** if show_transmission_details && hasItems */} + { + queueItems.map((queueItem, i) => ( + + {(() => { + if (queueItem.state === 'error') { + return ( +
    + { + gettext( + 'Error sending as {{name}} to {{destination}} at {{date}}', + { + name: () => {queueItem.unique_name}, + destination: queueItem.destination.name, + date: () => , + }, + ) + } +
    + ); + } else { + return ( +
    + { + gettext( + 'Sent/Queued as {{name}} to {{destination}} at {{date}}', + { + name: () => {queueItem.unique_name}, + destination: queueItem.destination.name, + date: () => , + }, + ) + } +
    + ); + } + })()} + + { + showModal(({closeModal}) => ( + + + {gettext('Item sent to Subscriber')} + + + + {tryShowingFormattedJson(queueItem.formatted_item)} + + + )); + }} + size="small" + /> +
    + )) + } +
    + ); + } +} diff --git a/scripts/apps/authoring-react/article-widgets/versions-and-item-history/versions-tab.tsx b/scripts/apps/authoring-react/article-widgets/versions-and-item-history/versions-tab.tsx new file mode 100644 index 0000000000..962be36afe --- /dev/null +++ b/scripts/apps/authoring-react/article-widgets/versions-and-item-history/versions-tab.tsx @@ -0,0 +1,291 @@ +import React from 'react'; +import {uniq} from 'lodash'; +import { + IArticle, + IExtensionActivationResult, + IRestApiResponse, + IDesk, + IStage, +} from 'superdesk-api'; +import {gettext, getItemLabel} from 'core/utils'; +import {httpRequestJsonLocal} from 'core/helpers/network'; +import {Card} from 'core/ui/components/Card'; +import {TimeElem} from 'apps/search/components'; +import {Spacer, SpacerBlock} from 'core/ui/components/Spacer'; +import {store} from 'core/data'; +import {StateComponent} from 'apps/search/components/fields/state'; +import {Button, ToggleBox} from 'superdesk-ui-framework/react'; +import {notNullOrUndefined} from 'core/helpers/typescript-helpers'; +import {Map} from 'immutable'; +import {sdApi} from 'api'; +import {dispatchInternalEvent} from 'core/internal-events'; +import {omitFields} from '../../data-layer'; +import {compareAuthoringEntities} from '../../compare-articles/compare-articles'; +import {previewAuthoringEntity} from '../../preview-article-modal'; +import {getArticleAdapter} from '../../article-adapter'; +import {SelectFilterable} from 'core/ui/components/select-filterable'; + +const loadingState: IState = { + versions: 'loading', + desks: Map(), + stages: Map(), + selectedForComparison: {from: null, to: null}, +}; + +type IProps = React.ComponentProps< + IExtensionActivationResult['contributions']['authoringSideWidgets'][0]['component'] +>; + +interface IState { + versions: Array | 'loading'; + desks: Map; + stages: Map; + selectedForComparison?: {from: IArticle; to: IArticle}; +} + +export class VersionsTab extends React.PureComponent { + constructor(props: IProps) { + super(props); + + this.state = loadingState; + + this.initialize = this.initialize.bind(this); + this.revert = this.revert.bind(this); + this.compareVersions = this.compareVersions.bind(this); + } + + initialize() { + Promise.all([ + httpRequestJsonLocal>({ + method: 'GET', + path: `/archive/${this.props.article._id}?version=all`, + }), + getArticleAdapter(), + ]).then(([res, adapter]) => { + const items = res._items.map((item) => adapter.toAuthoringReact(item)); + const itemsReversed = items.reverse(); + + const deskIds = uniq(items.map((item) => item.task?.desk).filter(notNullOrUndefined)); + const stageIds = uniq(items.map((item) => item.task?.stage).filter(notNullOrUndefined)); + + return Promise.all([ + httpRequestJsonLocal>({ + method: 'GET', + path: '/desks', + urlParams: {$in: deskIds}}, + ), + httpRequestJsonLocal>({ + method: 'GET', + path: '/stages', + urlParams: {$in: stageIds}}, + ), + ]).then(([resDesks, resStages]) => { + this.setState({ + versions: itemsReversed, + desks: Map(resDesks._items.map((item) => [item._id, item])), + stages: Map(resStages._items.map((item) => [item._id, item])), + selectedForComparison: { + from: itemsReversed[1], + to: itemsReversed[0], + }, + }); + }); + }); + } + + revert(version: IArticle) { + this.props.handleUnsavedChanges().then(({_id, _etag}) => { + httpRequestJsonLocal({ + method: 'PATCH', + path: `/archive/${_id}`, + payload: omitFields(version, true), + headers: { + 'If-Match': _etag, + }, + }).then(() => { + dispatchInternalEvent('dangerouslyForceReloadAuthoring', undefined); + + this.setState(loadingState); + + this.initialize(); + }); + }); + } + + compareVersions() { + const {from, to} = this.state.selectedForComparison; + + compareAuthoringEntities({ + item1: { + label: gettext('version {{n}}', {n: from._current_version}), + entity: from, + }, + item2: { + label: gettext('version {{n}}', {n: to._current_version}), + entity: to, + }, + getLanguage: () => '', + authoringStorage: this.props.authoringStorage, + fieldsAdapter: this.props.fieldsAdapter, + storageAdapter: this.props.storageAdapter, + }); + } + + componentDidMount() { + this.initialize(); + } + + render() { + if (this.state.versions === 'loading') { + return null; + } + + const {versions, desks, stages, selectedForComparison} = this.state; + const {readOnly, contentProfile, fieldsData} = this.props; + + const userEntities = + store.getState().entities.users; + + return ( + + + + { + this.setState({ + selectedForComparison: { + ...this.state.selectedForComparison, + from: val, + }, + }); + }} + getLabel={(item) => gettext('version: {{n}}', {n: item._current_version})} + required + /> + + { + this.setState({ + selectedForComparison: { + ...this.state.selectedForComparison, + to: val, + }, + }); + }} + getLabel={(item) => gettext('version: {{n}}', {n: item._current_version})} + required + /> + + + + +
    +
    + + +
    + + { + versions.map((item, i) => { + const canRevert = i !== 0 && !readOnly && !sdApi.article.isPublished(item); + + return ( + + + + + { + gettext('by {{user}}', { + user: userEntities[item.version_creator].display_name, + }) + } + + + + + +
    + {getItemLabel(item)} +
    + + + + { + item.task.desk != null && ( + + {desks.get(item.task.desk).name} + {stages.get(item.task.stage).name} + + ) + } + + + + +
    + {gettext('version: {{n}}', {n: item._current_version})} +
    + +
    + +
    +
    + + + + +
    + + +
    +
    + + { + canRevert && ( +
    +
    + ) + } +
    + + + ); + }) + } + + ); + } +} diff --git a/scripts/apps/authoring-react/authoring-angular-integration.tsx b/scripts/apps/authoring-react/authoring-angular-integration.tsx new file mode 100644 index 0000000000..8ea008d02e --- /dev/null +++ b/scripts/apps/authoring-react/authoring-angular-integration.tsx @@ -0,0 +1,449 @@ +/* eslint-disable react/display-name */ +/* eslint-disable react/no-multi-comp */ +import {assertNever} from 'core/helpers/typescript-helpers'; +import {DeskAndStage} from './subcomponents/desk-and-stage'; +import {LockInfo} from './subcomponents/lock-info'; +import {Button, ButtonGroup, IconButton, NavButton, Popover} from 'superdesk-ui-framework/react'; +import { + IArticle, + ITopBarWidget, + IExposedFromAuthoring, + IAuthoringOptions, +} from 'superdesk-api'; +import {appConfig, extensions} from 'appConfig'; +import {ITEM_STATE} from 'apps/archive/constants'; +import React from 'react'; +import {gettext} from 'core/utils'; +import {sdApi} from 'api'; +import ng from 'core/services/ng'; +import {AuthoringIntegrationWrapper} from './authoring-integration-wrapper'; +import {MarkedDesks} from './toolbar/mark-for-desks/mark-for-desks-popover'; +import {WithPopover} from 'core/helpers/with-popover'; +import {HighlightsCardContent} from './toolbar/highlights-management'; +import {authoringStorageIArticle} from './data-layer'; +import { + IStateInteractiveActionsPanelHOC, + IActionsInteractiveActionsPanelHOC, +} from 'core/interactive-article-actions-panel/index-hoc'; +import {IArticleActionInteractive} from 'core/interactive-article-actions-panel/interfaces'; +import {dispatchInternalEvent} from 'core/internal-events'; +import {notify} from 'core/notify/notify'; + +export interface IProps { + itemId: IArticle['_id']; +} + +function onClose() { + ng.get('authoringWorkspace').close(); + ng.get('$rootScope').$applyAsync(); +} + +function getInlineToolbarActions(options: IExposedFromAuthoring): IAuthoringOptions { + const { + item, + hasUnsavedChanges, + handleUnsavedChanges, + save, + initiateClosing, + keepChangesAndClose, + stealLock, + } = options; + const itemState: ITEM_STATE = item.state; + + const saveButton: ITopBarWidget = { + group: 'end', + priority: 0.2, + component: () => ( +
    + ); + } +} diff --git a/scripts/apps/authoring-react/fields/date/difference.tsx b/scripts/apps/authoring-react/fields/date/difference.tsx new file mode 100644 index 0000000000..df0c01c265 --- /dev/null +++ b/scripts/apps/authoring-react/fields/date/difference.tsx @@ -0,0 +1,21 @@ +import {formatDate} from 'core/get-superdesk-api-implementation'; +import React from 'react'; +import {IDateFieldConfig, IDateValueOperational, IDifferenceComponentProps} from 'superdesk-api'; +import {DifferenceGeneric} from '../difference-generic'; + +type IProps = IDifferenceComponentProps; + +export class Difference extends React.PureComponent { + render() { + const {value1, value2} = this.props; + + return ( + item} + template={({item}) => {item}} + /> + ); + } +} diff --git a/scripts/apps/authoring-react/fields/date/editor.tsx b/scripts/apps/authoring-react/fields/date/editor.tsx new file mode 100644 index 0000000000..19e19d4a7d --- /dev/null +++ b/scripts/apps/authoring-react/fields/date/editor.tsx @@ -0,0 +1,51 @@ +import React from 'react'; +import moment from 'moment'; +import differenceInCalendarDays from 'date-fns/differenceInCalendarDays'; +import {DatePickerISO} from 'superdesk-ui-framework/react'; +import {IDateFieldConfig, IDateUserPreferences, IDateValueOperational, IEditorComponentProps} from 'superdesk-api'; +import {appConfig} from 'appConfig'; +import {dateToServerString} from 'core/get-superdesk-api-implementation'; +import {getLocaleForDatePicker} from 'core/helpers/ui-framework'; +import {gettext} from 'core/utils'; + +type IProps = IEditorComponentProps; + +export class Editor extends React.PureComponent { + render() { + const Container = this.props.container; + const {readOnly} = this.props; + + return ( + + { + if (dateString === '') { + this.props.onChange(''); + return; + } + + this.props.onChange( + dateToServerString(new Date(dateString)), + ); + }} + dateFormat={appConfig.view.dateformat} + locale={getLocaleForDatePicker(this.props.language)} + headerButtonBar={this.props.config?.shortcuts?.map(({label, value, term}) => { + return { + label, + days: differenceInCalendarDays( + moment().startOf('day').add(value, term).toDate(), + new Date(), + ), + }; + })} + disabled={readOnly} + /> + + ); + } +} diff --git a/scripts/apps/authoring-react/fields/date/index.tsx b/scripts/apps/authoring-react/fields/date/index.tsx new file mode 100644 index 0000000000..19e22ff266 --- /dev/null +++ b/scripts/apps/authoring-react/fields/date/index.tsx @@ -0,0 +1,28 @@ +import { + ICustomFieldType, IDateFieldConfig, IDateUserPreferences, IDateValueOperational, IDateValueStorage, +} from 'superdesk-api'; +import {gettext} from 'core/utils'; +import {Editor} from './editor'; +import {Preview} from './preview'; +import {Difference} from './difference'; +import {Config} from './config'; + +export const DATE_FIELD_ID = 'date'; + +export function getDateField() +: ICustomFieldType { + const field: ICustomFieldType = { + id: DATE_FIELD_ID, + label: gettext('Date (authoring-react)'), + editorComponent: Editor, + previewComponent: Preview, + + hasValue: (valueOperational) => valueOperational != null, + getEmptyValue: () => null, + + differenceComponent: Difference, + configComponent: Config, + }; + + return field; +} diff --git a/scripts/apps/authoring-react/fields/date/preview.tsx b/scripts/apps/authoring-react/fields/date/preview.tsx new file mode 100644 index 0000000000..50aaebb117 --- /dev/null +++ b/scripts/apps/authoring-react/fields/date/preview.tsx @@ -0,0 +1,17 @@ +import React from 'react'; +import {IDateFieldConfig, IDateValueOperational, IPreviewComponentProps} from 'superdesk-api'; +import {formatDate} from 'core/get-superdesk-api-implementation'; + +type IProps = IPreviewComponentProps; + +export class Preview extends React.PureComponent { + render() { + if (this.props.value == null) { + return null; + } + + return ( +
    {formatDate(new Date(this.props.value))}
    + ); + } +} diff --git a/scripts/apps/authoring-react/fields/dateline/difference.tsx b/scripts/apps/authoring-react/fields/dateline/difference.tsx new file mode 100644 index 0000000000..e6ed1e0bba --- /dev/null +++ b/scripts/apps/authoring-react/fields/dateline/difference.tsx @@ -0,0 +1,20 @@ +import React from 'react'; +import {ICommonFieldConfig, IDatelineValueOperational, IDifferenceComponentProps} from 'superdesk-api'; +import {DifferenceGeneric} from '../difference-generic'; + +type IProps = IDifferenceComponentProps; + +export class Difference extends React.PureComponent { + render() { + const {value1, value2} = this.props; + + return ( + JSON.stringify(item)} + template={({item}) => {item}} + /> + ); + } +} diff --git a/scripts/apps/authoring-react/fields/dateline/editor.tsx b/scripts/apps/authoring-react/fields/dateline/editor.tsx new file mode 100644 index 0000000000..426fca6d78 --- /dev/null +++ b/scripts/apps/authoring-react/fields/dateline/editor.tsx @@ -0,0 +1,97 @@ +import React from 'react'; +import ng from 'core/services/ng'; +import {DatePickerISO} from 'superdesk-ui-framework/react'; +import { + ICommonFieldConfig, + IDatelineUserPreferences, + IDatelineValueOperational, + IEditorComponentProps, +} from 'superdesk-api'; +import {dateToServerString} from 'core/get-superdesk-api-implementation'; +import {getLocaleForDatePicker} from 'core/helpers/ui-framework'; +import {MultiSelectTreeWithTemplate} from 'core/ui/components/MultiSelectTreeWithTemplate'; +import {Spacer} from 'core/ui/components/Spacer'; +import {appConfig} from 'appConfig'; +import {gettext} from 'core/utils'; + +type IProps = IEditorComponentProps< + IDatelineValueOperational, + ICommonFieldConfig, + IDatelineUserPreferences +>; + +type ICancelFn = () => void; + +function searchOptions( + term: string, + callback: (res: any) => void, +): ICancelFn { + const abortController = new AbortController(); + + ng.get('places').searchDateline(term, 'en', abortController.signal).then((res) => { + callback({ + nodes: res.slice(0, 10).map((item) => ({value: item})), + lookup: {}, + }); + }); + + return () => abortController.abort(); +} + +export class Editor extends React.PureComponent { + render() { + const Container = this.props.container; + + return ( + + + searchOptions(term, callback)} + values={[this.props.value?.located]} + onChange={([value]) => { + this.props.onChange({ + ...this.props.value, + located: value, + }); + }} + optionTemplate={ + ({item}) => item != null ? ( + + {item?.city}
    + {item?.state}, {item?.country} +
    + ) : null + } + valueTemplate={ + ({item}) => item != null ? ( + + {item?.city} + + ) : null + } + getId={(option) => option?.city_code} + getLabel={(item) => item?.city} + /> + + { + this.props.onChange({ + ...this.props.value, + date: dateToServerString(new Date(dateString)), + }); + }} + dateFormat={appConfig.view.dateformat} + locale={getLocaleForDatePicker(this.props.language)} + disabled={this.props.value?.located?.city == null} + /> +
    +
    + ); + } +} diff --git a/scripts/apps/authoring-react/fields/dateline/index.tsx b/scripts/apps/authoring-react/fields/dateline/index.tsx new file mode 100644 index 0000000000..f65bbc483f --- /dev/null +++ b/scripts/apps/authoring-react/fields/dateline/index.tsx @@ -0,0 +1,38 @@ +import { + ICommonFieldConfig, + ICustomFieldType, + IDatelineUserPreferences, + IDatelineValueOperational, + IDatelineValueStorage, +} from 'superdesk-api'; +import {gettext} from 'core/utils'; +import {Editor} from './editor'; +import {Preview} from './preview'; +import {Difference} from './difference'; + +export const DATELINE_FIELD_ID = 'dateline'; + +type DatelineFieldType = ICustomFieldType< + IDatelineValueOperational, + IDatelineValueStorage, + ICommonFieldConfig, + IDatelineUserPreferences +>; + +export function getDatelineField() +: DatelineFieldType { + const field: DatelineFieldType = { + id: DATELINE_FIELD_ID, + label: gettext('Dateline (authoring-react)'), + editorComponent: Editor, + previewComponent: Preview, + + hasValue: (valueOperational) => valueOperational != null, + getEmptyValue: () => null, + + differenceComponent: Difference, + configComponent: () => null, + }; + + return field; +} diff --git a/scripts/apps/authoring-react/fields/dateline/preview.tsx b/scripts/apps/authoring-react/fields/dateline/preview.tsx new file mode 100644 index 0000000000..d7b11ecd0b --- /dev/null +++ b/scripts/apps/authoring-react/fields/dateline/preview.tsx @@ -0,0 +1,20 @@ +import React from 'react'; +import { + IDatelineFieldConfig, + IDatelineValueOperational, + IPreviewComponentProps, +} from 'superdesk-api'; + +type IProps = IPreviewComponentProps; + +export class Preview extends React.PureComponent { + render() { + if (this.props.value == null) { + return null; + } + + return ( +
    {this.props.value}
    + ); + } +} diff --git a/scripts/apps/authoring-react/fields/difference-generic.tsx b/scripts/apps/authoring-react/fields/difference-generic.tsx new file mode 100644 index 0000000000..c42b81d02e --- /dev/null +++ b/scripts/apps/authoring-react/fields/difference-generic.tsx @@ -0,0 +1,114 @@ +import {SpacerInlineFlex} from 'core/ui/components/Spacer'; +import {groupBy, keyBy} from 'lodash'; +import React from 'react'; + +interface IProps { + items1: Array; + items2: Array; + getId(item: T): string; + template: React.ComponentType<{item: T}>; +} + +type IChangeType = 'removal' | 'addition' | 'order-change'; + +interface IChange { + type: IChangeType; + id: string; + item: T; +} + +export class DifferenceGeneric extends React.PureComponent> { + render() { + const {items1, items2, getId} = this.props; + const Template = this.props.template; + + const items1Lookup = keyBy(items1, getId); + const items2Lookup = keyBy(items2, getId); + + const result: Array> = []; + + for (const item of items2) { + const id = getId(item); + + if (id in items1Lookup !== true) { + result.push({ + id: id, + item: item, + type: 'addition', + }); + } + } + + for (const item of items1) { + const id = getId(item); + + if (items2Lookup.hasOwnProperty(id) !== true) { + result.push({ + id: id, + item: item, + type: 'removal', + }); + } + } + + const resultLookup = groupBy(result, ({type}) => type); + + const changedItemIds = new Set(result.map(({id}) => id)); + const unChangedItems = items2.filter((item) => { + const id = getId(item); + + return changedItemIds.has(id) !== true; + }); + + const removed = resultLookup['removal'] ?? []; + const added = resultLookup['addition'] ?? []; + + return ( + + { + removed.length > 0 && ( + + { + removed.map((change, i) => ( + + +