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

[UI Framework] Add isReadOnly prop to KuiCodeEditor. #14804

Merged
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
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
15 changes: 15 additions & 0 deletions ui_framework/doc_site/src/views/code_editor/code_editor_example.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,9 @@ import {
import CodeEditor from './code_editor';
const codeEditorSource = require('!!raw-loader!./code_editor');

import ReadOnly from './read_only';
const readOnlySource = require('!!raw-loader!./read_only');

export default props => (
<GuidePage title={props.route.name}>
<GuideSection
Expand All @@ -36,5 +39,17 @@ export default props => (
<CodeEditor />
</GuideDemo>
</GuideSection>

<GuideSection
title="Read-only"
source={[{
type: GuideSectionTypes.JS,
code: readOnlySource,
}]}
>
<GuideDemo>
<ReadOnly />
</GuideDemo>
</GuideSection>
</GuidePage>
);
27 changes: 27 additions & 0 deletions ui_framework/doc_site/src/views/code_editor/read_only.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
import React, { Component } from 'react';

import 'brace/mode/less';
import 'brace/theme/github';

import {
KuiCodeEditor
} from '../../../../components';

export default class extends Component {
state = {
value: '<p>This code is read only</p>'
};

render() {
return (
<KuiCodeEditor
mode="less"
theme="github"
width="100%"
value={this.state.value}
setOptions={{ fontSize: '14px' }}
isReadOnly
/>
);
}
}
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP

exports[`KuiCodeEditor hint element should be disabled when the ui ace box gains focus 1`] = `
exports[`KuiCodeEditor behavior hint element should be disabled when the ui ace box gains focus 1`] = `
<div
className="kuiCodeEditorKeyboardHint kuiCodeEditorKeyboardHint-isInactive"
data-test-subj="codeEditorHint"
Expand All @@ -23,7 +23,7 @@ exports[`KuiCodeEditor hint element should be disabled when the ui ace box gains
</div>
`;

exports[`KuiCodeEditor hint element should be enabled when the ui ace box loses focus 1`] = `
exports[`KuiCodeEditor behavior hint element should be enabled when the ui ace box loses focus 1`] = `
<div
className="kuiCodeEditorKeyboardHint"
data-test-subj="codeEditorHint"
Expand Down Expand Up @@ -158,3 +158,99 @@ exports[`KuiCodeEditor is rendered 1`] = `
</div>
</div>
`;

exports[`KuiCodeEditor props isReadOnly doesn't render hint 1`] = `
<div
class="kuiCodeEditorWrapper"
>
<div
class=" ace_editor ace-tm"
id="brace-editor"
style="width: 500px; height: 500px;"
>
<textarea
autocapitalize="off"
autocorrect="off"
class="ace_text-input"
spellcheck="false"
style="opacity: 0;"
tabindex="-1"
wrap="off"
/>
<div
class="ace_gutter"
>
<div
class="ace_layer ace_gutter-layer ace_folding-enabled"
/>
<div
class="ace_gutter-active-line"
style="display: none;"
/>
</div>
<div
class="ace_scroller"
>
<div
class="ace_content"
>
<div
class="ace_layer ace_print-margin-layer"
>
<div
class="ace_print-margin"
style="left: 4px; visibility: visible;"
/>
</div>
<div
class="ace_layer ace_marker-layer"
/>
<div
class="ace_layer ace_text-layer"
style="padding: 0px 4px;"
/>
<div
class="ace_layer ace_marker-layer"
/>
<div
class="ace_layer ace_cursor-layer ace_hidden-cursors"
>
<div
class="ace_cursor"
/>
</div>
</div>
</div>
<div
class="ace_scrollbar ace_scrollbar-v"
style="display: none; width: 20px;"
>
<div
class="ace_scrollbar-inner"
style="width: 20px;"
/>
</div>
<div
class="ace_scrollbar ace_scrollbar-h"
style="display: none; height: 20px;"
>
<div
class="ace_scrollbar-inner"
style="height: 20px;"
/>
</div>
<div
style="height: auto; width: auto; top: 0px; left: 0px; visibility: hidden; position: absolute; white-space: pre; overflow: hidden;"
>
<div
style="height: auto; width: auto; top: 0px; left: 0px; visibility: hidden; position: absolute; white-space: pre; overflow: visible;"
/>
<div
style="height: auto; width: auto; top: 0px; left: 0px; visibility: hidden; position: absolute; white-space: pre; overflow: visible;"
>
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
</div>
</div>
</div>
</div>
`;
35 changes: 30 additions & 5 deletions ui_framework/src/components/code_editor/code_editor.js
Original file line number Diff line number Diff line change
Expand Up @@ -58,18 +58,27 @@ export class KuiCodeEditor extends Component {
width,
height,
onBlur, // eslint-disable-line no-unused-vars
isReadOnly,
setOptions,
...rest,
} = this.props;

const classes = classNames('kuiCodeEditorKeyboardHint', {
'kuiCodeEditorKeyboardHint-isInactive': !this.state.isHintActive
});

return (
<div
className="kuiCodeEditorWrapper"
style={{ width, height }}
>
let prompt;

const options = { ...setOptions };

if (isReadOnly) {
Object.assign(options, {
readOnly: true,
highlightActiveLine: false,
highlightGutterLine: false,
});
} else {
prompt = (
<div
className={classes}
id={this.idGenerator('codeEditor')}
Expand All @@ -88,12 +97,22 @@ export class KuiCodeEditor extends Component {
When you&rsquo;re done, press Escape to stop editing.
</p>
</div>
);
}

return (
<div
className="kuiCodeEditorWrapper"
style={{ width, height }}
>
{prompt}

<AceEditor
ref={this.aceEditorRef}
width={width}
height={height}
onBlur={this.onBlurAce}
setOptions={options}
{...rest}
/>
</div>
Expand All @@ -105,4 +124,10 @@ KuiCodeEditor.propTypes = {
width: PropTypes.string,
height: PropTypes.string,
onBlur: PropTypes.func,
isReadOnly: PropTypes.bool,
setOptions: PropTypes.object,
};

KuiCodeEditor.defaultProps = {
setOptions: {},
};
81 changes: 46 additions & 35 deletions ui_framework/src/components/code_editor/code_editor.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,53 +14,64 @@ jest.mock('../../services/accessibility/html_id_generator', () => ({
}));

describe('KuiCodeEditor', () => {
let element;

beforeEach(() => {
element = mount(<KuiCodeEditor/>);
});

test('is rendered', () => {
const component = mount(<KuiCodeEditor {...requiredProps}/>);
const component = mount(<KuiCodeEditor {...requiredProps} />);
expect(takeMountedSnapshot(component)).toMatchSnapshot();
});

describe('hint element', () => {
test('should be tabable', () => {
expect(element.find('[data-test-subj="codeEditorHint"]').prop('tabIndex')).toBe('0');
describe('props', () => {
describe('isReadOnly', () => {
test(`doesn't render hint`, () => {
const component = mount(<KuiCodeEditor isReadOnly />);
expect(takeMountedSnapshot(component)).toMatchSnapshot();
});
});
});

test('should be disabled when the ui ace box gains focus', () => {
const hint = element.find('[data-test-subj="codeEditorHint"]');
hint.simulate('keydown', { keyCode: keyCodes.ENTER });
expect(hint).toMatchSnapshot();
});
describe('behavior', () => {
let component;

test('should be enabled when the ui ace box loses focus', () => {
const hint = element.find('[data-test-subj="codeEditorHint"]');
hint.simulate('keydown', { keyCode: keyCodes.ENTER });
element.instance().onBlurAce();
expect(hint).toMatchSnapshot();
beforeEach(() => {
component = mount(<KuiCodeEditor/>);
});
});

describe('interaction', () => {
test('bluring the ace textbox should call a passed onBlur prop', () => {
const blurSpy = sinon.spy();
const el = mount(<KuiCodeEditor onBlur={blurSpy}/>);
el.instance().onBlurAce();
expect(blurSpy.called).toBe(true);
describe('hint element', () => {
test('should be tabable', () => {
expect(component.find('[data-test-subj="codeEditorHint"]').prop('tabIndex')).toBe('0');
});

test('should be disabled when the ui ace box gains focus', () => {
const hint = component.find('[data-test-subj="codeEditorHint"]');
hint.simulate('keydown', { keyCode: keyCodes.ENTER });
expect(hint).toMatchSnapshot();
});

test('should be enabled when the ui ace box loses focus', () => {
const hint = component.find('[data-test-subj="codeEditorHint"]');
hint.simulate('keydown', { keyCode: keyCodes.ENTER });
component.instance().onBlurAce();
expect(hint).toMatchSnapshot();
});
});

test('pressing escape in ace textbox will enable overlay', () => {
element.instance().onKeydownAce({
preventDefault: () => {},
stopPropagation: () => {},
keyCode: keyCodes.ESCAPE
describe('interaction', () => {
test('bluring the ace textbox should call a passed onBlur prop', () => {
const blurSpy = sinon.spy();
const el = mount(<KuiCodeEditor onBlur={blurSpy}/>);
el.instance().onBlurAce();
expect(blurSpy.called).toBe(true);
});

test('pressing escape in ace textbox will enable overlay', () => {
component.instance().onKeydownAce({
preventDefault: () => {},
stopPropagation: () => {},
keyCode: keyCodes.ESCAPE
});
expect(
component.find('[data-test-subj="codeEditorHint"]').matchesElement(document.activeElement)
).toBe(true);
});
expect(
element.find('[data-test-subj="codeEditorHint"]').matchesElement(document.activeElement)
).toBe(true);
});
});
});