Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Read only mode #1035

Merged
merged 107 commits into from
Sep 29, 2020
Merged
Show file tree
Hide file tree
Changes from 79 commits
Commits
Show all changes
107 commits
Select commit Hold shift + click to select a range
e78b22d
init readonly module
hboylan Feb 12, 2020
e40c591
expose toggleReadOnly method in public api
hboylan Feb 12, 2020
2a32659
alias readonly api on editor
hboylan Feb 13, 2020
b710e5e
include readonly toggle button in dev example
hboylan Feb 13, 2020
d259339
unbind ui events in readonly mode
hboylan Feb 13, 2020
0586a81
switch to setReadOnly method in ui and paste modules
hboylan Feb 13, 2020
cb2622e
reset paste module in readonly mode
hboylan Feb 13, 2020
fac51fd
update internal prepare methods with setReadOnly
hboylan Feb 14, 2020
7b1cffe
destroy mutation observer when read-only enabled
hboylan Feb 14, 2020
f2b4018
rename readonly toggle methods
hboylan Feb 16, 2020
1a4fff1
unbind events
khaydarov Apr 11, 2020
f5e9a66
[Improvements] ESLint action (#1099)
neSpecc Apr 11, 2020
7c726d4
generate id on event listeners
khaydarov Apr 11, 2020
552d090
Merge remote-tracking branch 'origin/release/2.18' into pr/1035
khaydarov Apr 11, 2020
4c0d806
[Refactor] ESLint fixed (#1100)
gohabereg Apr 18, 2020
d42042d
save
khaydarov Apr 20, 2020
1a6f9a1
Merge remote-tracking branch 'origin/release/2.18' into pr/1035
khaydarov Apr 20, 2020
ffa78fd
resolve conflicts
khaydarov Apr 20, 2020
e671a6e
fix event handlers
khaydarov Apr 20, 2020
0680c4d
add read only flag everywhere
khaydarov Apr 20, 2020
21cac86
[Feature] i18n (#1106)
neSpecc Apr 20, 2020
b32b2cb
Merge remote-tracking branch 'origin/release/2.18' into pr/1035
khaydarov Apr 20, 2020
968280a
resolve conflicts
khaydarov Apr 20, 2020
775ccbc
Remove unused submodule
tasuku-s Apr 23, 2020
f4c4051
Fixed: icon centering in Firefox
neSpecc Apr 25, 2020
2997ed0
Do not load styles twice (#1112)
gohabereg Apr 25, 2020
c66105b
ready for testing
khaydarov Apr 25, 2020
4a81699
Show warning if Block to delete is not found (#1111)
gohabereg Apr 25, 2020
7eb642d
Save Tools' order in the Toolbox (#1113)
gohabereg Apr 25, 2020
72213f2
fix $.isEmpty performance (#1096)
tasuku-s Apr 25, 2020
7f876e8
Add issue templates (#1114)
gohabereg Apr 25, 2020
8456da7
requested changes
khaydarov Apr 25, 2020
66e70fa
Update issue templates (#1121)
neSpecc Apr 25, 2020
5941ef0
ready for testing. hurr-ah
khaydarov Apr 25, 2020
9b92e6c
Merge remote-tracking branch 'origin/release/2.18' into pr/1035
khaydarov Apr 25, 2020
77534fe
update changelog
khaydarov Apr 25, 2020
4f6164c
fix copy and cut
khaydarov Apr 25, 2020
5218a6a
Allowing deleting block by block id (#1108)
athul7744 Apr 25, 2020
ae8591c
Allow navigate next from last non-initial block (#1110)
gohabereg Apr 25, 2020
c3ba372
update
khaydarov Apr 26, 2020
f291aa0
Merge branch 'release/2.18' into pr/1035
khaydarov Apr 26, 2020
ec63dcd
update
khaydarov Apr 26, 2020
6edb7bc
fix eslint problems
khaydarov Apr 26, 2020
05766a6
update
khaydarov Apr 26, 2020
65f51c2
fix paste
khaydarov Apr 26, 2020
b90b8c5
update
khaydarov Apr 28, 2020
6e79df6
upgrade
khaydarov Apr 30, 2020
b90532e
update
khaydarov Apr 30, 2020
f6b7b2e
Create CODE_OF_CONDUCT.md (#1171)
talyguryn May 27, 2020
e3410be
Update dependencies (#1122)
gohabereg May 27, 2020
b0ddb10
Feature/disable tab event config (#1164)
flaming-cl May 27, 2020
1eb6d81
Highlight first block on autofocus (#1127)
gohabereg May 27, 2020
5fe8b08
Fix shortcut for external tools (#1141)
sonnn May 27, 2020
7c3bf76
Hotfix/issue1133 selection shortcut removed on editor destroy (#1140)
sis-dk May 27, 2020
ffe5bbc
[Feature] BlockAPI Interface (#1075)
gohabereg May 27, 2020
ff6bd2d
Fix BlockManager.insert method (#1172)
gohabereg Jun 2, 2020
e1500f7
Update tools master branches (#1180)
talyguryn Jun 2, 2020
26b19a3
Fix behaviour of inputs editing in block settings (#1123)
gohabereg Jun 2, 2020
8341c33
Merge branch 'master' into release/2.18
neSpecc Jun 2, 2020
239aaf6
lint code
neSpecc Jun 3, 2020
4cea66f
Update CHANGELOG.md
neSpecc Jun 3, 2020
0e374ab
test commit
khaydarov Jun 4, 2020
7992995
Revert "test commit"
khaydarov Jun 4, 2020
cefe8da
Merge branch 'release/2.18' into pr/1035
khaydarov Jun 4, 2020
9bbf37d
last updates
khaydarov Jun 5, 2020
884baf2
comment
khaydarov Jun 9, 2020
f983a5a
requested changes
khaydarov Jun 9, 2020
d7d8be8
eslint fixes
khaydarov Jun 9, 2020
f07886d
last changes
khaydarov Jul 26, 2020
c82d359
Merge branch 'next' into pr/1035
khaydarov Jul 26, 2020
ea75bf2
small fix
khaydarov Jul 26, 2020
5772c04
fix defect with native selection
khaydarov Jul 26, 2020
96a1780
type magic
khaydarov Jul 26, 2020
22dd508
update
khaydarov Jul 26, 2020
a296077
updates
khaydarov Jul 26, 2020
8f74b37
eslint fixes
khaydarov Jul 26, 2020
4aaddfe
Check tools compatibility on read-only toggle
gohabereg Aug 22, 2020
1a61d3f
Merge branch 'next' of github.com:codex-team/editor.js into pr/1035
gohabereg Aug 22, 2020
a00090a
Update submodules
gohabereg Aug 22, 2020
99d074a
Use Math.floor instead of ~~
gohabereg Aug 23, 2020
014ea48
Merge branch 'next' into pr/1035
neSpecc Aug 26, 2020
f02e49c
upd demo page toggler
neSpecc Aug 26, 2020
93829a0
rm unused styles from example
neSpecc Aug 26, 2020
5921260
add description to generic type
neSpecc Aug 26, 2020
6ec8d06
Saver API: return promise.reject instead of undifined
neSpecc Aug 26, 2020
1af17f0
update
khaydarov Aug 26, 2020
e8efd87
update nodes
khaydarov Aug 26, 2020
edece25
lint fixes
khaydarov Aug 26, 2020
f0b5dc0
save changes
khaydarov Aug 27, 2020
189e7b6
Merge branch 'next' into pr/1035
khaydarov Aug 28, 2020
34a425f
Create editor.js.map
khaydarov Aug 28, 2020
7a3162f
updates
khaydarov Aug 28, 2020
748aa7d
save changes
khaydarov Aug 28, 2020
6977b33
Merge branch 'next' into pr/1035
khaydarov Sep 1, 2020
ef17dbf
Update editor.js.map
khaydarov Sep 1, 2020
a318dff
update
khaydarov Sep 1, 2020
b2865ad
new bundle
khaydarov Sep 1, 2020
15b2794
Update methods call prevention in read-only mode
gohabereg Sep 13, 2020
d04f1ae
Remove API restrictions for read only mode
gohabereg Sep 19, 2020
7f1ffcd
Update CBS event handlerr
gohabereg Sep 21, 2020
efef57f
Merge branch 'next' of github.com:codex-team/editor.js into pr/1035
gohabereg Sep 22, 2020
cd6d263
rm duplicated code
neSpecc Sep 29, 2020
899dd3f
Optimiza isAnyBlockSelected method
gohabereg Sep 29, 2020
59b2688
Update bundle
gohabereg Sep 29, 2020
24ab8f9
Update src/components/modules/blockSelection.ts
neSpecc Sep 29, 2020
d671342
Merge branch 'next' into pr/1035
neSpecc Sep 29, 2020
29945ae
pull next
neSpecc Sep 29, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion dist/editor.js

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion docs/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

### 2.19

- `New` - Read-only mode 🥳 [#837](https://github.com/codex-team/editor.js/issues/837)
- `Fix` - Fixed issue with `editor.blocks.delete(index)` method which throws an error when Editor.js is not focused, even after providing a valid index. [#1182](https://github.com/codex-team/editor.js/issues/1182)
- `Improvements` - The `initialBlock` property of Editor config is deprecated. Use the `defaultBlock` instead. [#993](https://github.com/codex-team/editor.js/issues/993)
- `Fix` - Fixed the issue of toolbar not disappearing on entering input in Chinese, Hindi and some other languages. [#1196](https://github.com/codex-team/editor.js/issues/1196)
Expand Down Expand Up @@ -31,7 +32,6 @@

> *Breaking changes* `blocks.getBlockByIndex` method now returns BlockAPI object. To access old value, use BlockAPI.holder property


### 2.17

- `Improvements` - Editor's [onchange callback](https://editorjs.io/configuration#editor-modifications-callback) now accepts an API as a parameter
Expand Down
10 changes: 7 additions & 3 deletions example/assets/demo.css
Original file line number Diff line number Diff line change
Expand Up @@ -87,10 +87,14 @@ body {
display: none;
}

.ce-example__buttons {
display: flex;
flex-direction: column;
}

.ce-example__button {
display: block;
margin: 50px auto;
max-width: 180px;
display: inline-flex;
margin: 20px auto;
background: #4A9DF8;
padding: 17px 30px;
box-shadow: 0 22px 18px -4px rgba(137, 207, 255, 0.77);
Expand Down
42 changes: 38 additions & 4 deletions example/example-dev.html
Original file line number Diff line number Diff line change
Expand Up @@ -31,8 +31,13 @@
<div id="hint" style="text-align: center;">
No submodules found. Run <code class="inline-code">yarn pull_tools</code>
</div>
<div class="ce-example__button" id="saveButton">
editor.save()
<div class="ce-example__buttons">
<div class="ce-example__button" id="saveButton">
editor.save()
</div>
<div class="ce-example__button" id="toggleReadOnlyButton">
editor.toggleReadOnly()
</div>
</div>
</div>
<div class="ce-example__output">
Expand Down Expand Up @@ -79,11 +84,23 @@
*/
const saveButton = document.getElementById('saveButton');

/**
* Toggle read-only button
*/
const toggleReadOnlyButton = document.getElementById('toggleReadOnlyButton');

/**
* To initialize the Editor, create a new instance with configuration object
* @see docs/installation.md for mode details
*/
const readOnly = true;

var editor = new EditorJS({
/**
* Enable read only
*/
readOnly: readOnly,

/**
* Wrapper of Editor
*/
Expand Down Expand Up @@ -272,7 +289,9 @@
]
},
onReady: function(){
saveButton.click();
if (!readOnly) {
saveButton.click();
}
},
onChange: function() {
console.log('something changed');
Expand All @@ -284,10 +303,25 @@
* Saving example
*/
saveButton.addEventListener('click', function () {
editor.save().then((savedData) => {
const promise = editor.save();

if (!promise) {
return;
}

promise.then((savedData) => {
cPreview.show(savedData, document.getElementById("output"));
});
});

/**
* Toggle read-only example
*/
toggleReadOnlyButton.addEventListener('click', async () => {
const readOnlyState = await editor.readonly.toggle();

console.log('Read only', readOnlyState ? 'enabled' : 'disabled');
});
</script>
</body>
</html>
2 changes: 1 addition & 1 deletion example/tools/checklist
Submodule checklist updated 3 files
+1 −1 dist/bundle.js
+1 −1 package.json
+57 −25 src/index.js
2 changes: 1 addition & 1 deletion example/tools/code
Submodule code updated 3 files
+1 −1 dist/bundle.js
+1 −1 package.json
+17 −1 src/index.js
2 changes: 1 addition & 1 deletion example/tools/delimiter
2 changes: 1 addition & 1 deletion example/tools/embed
Submodule embed updated 4 files
+1 −1 dist/bundle.js
+1 −1 package.json
+53 −37 src/index.js
+12 −220 yarn.lock
2 changes: 1 addition & 1 deletion example/tools/header
Submodule header updated 3 files
+2 −2 dist/bundle.js
+1 −1 package.json
+14 −2 src/index.js
2 changes: 1 addition & 1 deletion example/tools/image
Submodule image updated 6 files
+18 −0 README.md
+2 −2 dist/bundle.js
+1 −1 package.json
+15 −1 src/index.js
+15 −5 src/tunes.js
+4 −2 src/ui.js
2 changes: 1 addition & 1 deletion example/tools/link
Submodule link updated 3 files
+1 −1 dist/bundle.js
+1 −1 package.json
+76 −46 src/index.js
2 changes: 1 addition & 1 deletion example/tools/list
Submodule list updated from 497c04 to 439e5f
2 changes: 1 addition & 1 deletion example/tools/quote
Submodule quote updated 3 files
+1 −1 dist/bundle.js
+1 −1 package.json
+43 −29 src/index.js
2 changes: 1 addition & 1 deletion example/tools/raw
Submodule raw updated 3 files
+1 −1 dist/bundle.js
+1 −1 package.json
+39 −16 src/index.js
2 changes: 1 addition & 1 deletion example/tools/simple-image
2 changes: 1 addition & 1 deletion example/tools/table
2 changes: 1 addition & 1 deletion example/tools/warning
Submodule warning updated 3 files
+1 −1 dist/bundle.js
+1 −1 package.json
+42 −31 src/index.js
66 changes: 65 additions & 1 deletion src/components/__module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@ import { EditorModules } from '../types-internal/editor-modules';
import { EditorConfig } from '../../types';
import { ModuleConfig } from '../types-internal/module-config';

export type ModuleNodes = {[key: string]: HTMLElement};

/**
* @abstract
* @class Module
Expand All @@ -11,7 +13,12 @@ import { ModuleConfig } from '../types-internal/module-config';
* @property {object} config - Editor user settings
* @property {EditorModules} Editor - List of Editor modules
*/
export default class Module {
export default class Module<T extends ModuleNodes = {}> {
/**
* Each module can provide some UI elements that will be stored in this property
*/
public nodes: T;

/**
* Editor modules list
*
Expand All @@ -26,6 +33,50 @@ export default class Module {
*/
protected config: EditorConfig;

/**
* This object provides methods to push into set of listeners that being dropped when read-only mode is enabled
*/
protected readOnlyMutableListeners = {
/**
* Assigns event listener on DOM element and pushes into special array that might be removed
*
* @param {EventTarget} element - DOM Element
* @param {string} eventType - Event name
* @param {Function} handler - Event handler
* @param {boolean|AddEventListenerOptions} options - Listening options
*/
on: (
neSpecc marked this conversation as resolved.
Show resolved Hide resolved
element: EventTarget,
eventType: string,
handler: (event: Event) => void,
options: boolean | AddEventListenerOptions = false
): void => {
neSpecc marked this conversation as resolved.
Show resolved Hide resolved
const { Listeners } = this.Editor;

this.mutableListenerIds.push(
Listeners.on(element, eventType, handler, options)
);
},

/**
* Clears all mutable listeners
*/
clearAll: (): void => {
khaydarov marked this conversation as resolved.
Show resolved Hide resolved
const { Listeners } = this.Editor;

for (const id of this.mutableListenerIds) {
Listeners.offById(id);
}

this.mutableListenerIds = [];
},
};

/**
* The set of listener identifiers which will be dropped in read-only mode
*/
private mutableListenerIds: string[] = [];

/**
* @class
* @param {EditorConfig} config - Editor's config
Expand All @@ -46,4 +97,17 @@ export default class Module {
public set state(Editor: EditorModules) {
this.Editor = Editor;
}

/**
* Remove memorized nodes
*/
public removeAllNodes(): void {
for (const key in this.nodes) {
const node = this.nodes[key];

if (node instanceof HTMLElement) {
node.remove();
khaydarov marked this conversation as resolved.
Show resolved Hide resolved
}
}
}
}
8 changes: 8 additions & 0 deletions src/components/block/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,11 @@ interface BlockConstructorOptions {
* Editor's API methods
*/
api: ApiModule;
neSpecc marked this conversation as resolved.
Show resolved Hide resolved

/**
* This flag indicates that the Block should be constructed in the read-only mode.
*/
readOnly: boolean;
}

/**
Expand Down Expand Up @@ -199,13 +204,15 @@ export default class Block {
* @param {BlockToolConstructable} options.Tool — Tool's class
* @param {ToolSettings} options.settings - default tool's config
* @param {ApiModule} options.api - Editor API module for pass it to the Block Tunes
* @param {boolean} options.readOnly - Read-Only flag
*/
constructor({
name,
data,
Tool,
settings,
api,
readOnly,
}: BlockConstructorOptions) {
this.name = name;
this.class = Tool;
Expand All @@ -221,6 +228,7 @@ export default class Block {
config: this.config,
api: this.api.getMethodsForTool(name, ToolType.Block),
block: this.blockAPI,
readOnly: readOnly,
});

this.holder = this.compose();
Expand Down
11 changes: 7 additions & 4 deletions src/components/core.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,5 @@
import $ from './dom';
// eslint-disable-next-line import/no-duplicates
import * as _ from './utils';
// eslint-disable-next-line import/no-duplicates
import { LogLevels } from './utils';
import { EditorConfig, OutputData, SanitizerConfig } from '../../types';
import { EditorModules } from '../types-internal/editor-modules';
import I18n from './i18n';
Expand Down Expand Up @@ -151,7 +148,7 @@ export default class Core {
}

if (!this.config.logLevel) {
this.config.logLevel = LogLevels.VERBOSE;
this.config.logLevel = _.LogLevels.VERBOSE;
}

_.setLogLevel(this.config.logLevel);
Expand Down Expand Up @@ -209,6 +206,8 @@ export default class Core {
}
}

this.config.readOnly = this.config.readOnly as boolean || false;

/**
* Adjust i18n
*/
Expand Down Expand Up @@ -277,13 +276,17 @@ export default class Core {
public async start(): Promise<void> {
const modulesToPrepare = [
'Tools',
'ReadOnly',
'UI',
'Toolbar',
'InlineToolbar',
'BlockManager',
'Paste',
'DragNDrop',
'ModificationsObserver',
'BlockSelection',
'RectangleSelection',
'CrossBlockSelection',
];

await modulesToPrepare.reduce(
Expand Down
1 change: 1 addition & 0 deletions src/components/modules/api/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ export default class API extends Module {
inlineToolbar: this.Editor.InlineToolbarAPI.methods,
tooltip: this.Editor.TooltipAPI.methods,
i18n: this.Editor.I18nAPI.methods,
readonly: this.Editor.ReadOnlyAPI.methods,
} as APIInterfaces;
}

Expand Down
28 changes: 28 additions & 0 deletions src/components/modules/api/readonly.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
import Module from '../../__module';
import { ReadOnly } from '../../../../types/api';

/**
* @class ReadOnlyAPI
* @classdesc ReadOnly API
*/
export default class ReadOnlyAPI extends Module {
/**
* Available methods
*/
public get methods(): ReadOnly {
return {
toggle: (state): Promise<boolean> => this.toggle(state),
};
}

/**
* Set or toggle read-only state
*
* @param {boolean|undefined} state - set or toggle state
*
* @returns {boolean} current value
*/
public toggle(state?: boolean): Promise<boolean> {
return this.Editor.ReadOnly.toggle(state);
}
}
11 changes: 10 additions & 1 deletion src/components/modules/api/saver.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import Module from '../../__module';
import { Saver } from '../../../../types/api';
import { OutputData } from '../../../../types';
import * as _ from '../../utils';

/**
* @class SaverAPI
Expand All @@ -20,8 +21,16 @@ export default class SaverAPI extends Module {

/**
* Return Editor's data
*
* @returns {OutputData | undefined}
*/
public save(): Promise<OutputData> {
public save(): Promise<OutputData> | undefined {
if (this.Editor.ReadOnly.isEnabled) {
_.logLabeled('Editor\'s content can not be saved in read-only mode', 'warn');

return;
}

return this.Editor.Saver.save();
}
}
Loading