Added more keyboard options, testing some CTRL+ENTER

This commit is contained in:
2018-07-09 08:41:50 +10:00
parent d0b4e453a3
commit 17a68aeaf1
5 changed files with 134 additions and 25 deletions

View File

@ -28,6 +28,7 @@ import ReactDOM from 'react-dom';
import { createStore, applyMiddleware } from 'redux'; import { createStore, applyMiddleware } from 'redux';
import { Provider } from 'react-redux'; import { Provider } from 'react-redux';
import RootReducer from './reducers/RootReducer' import RootReducer from './reducers/RootReducer'
import Keyboard from './keyboard/Keyboard';
//Import Stylesheet //Import Stylesheet
import Styles from './styles/index'; import Styles from './styles/index';
@ -41,6 +42,9 @@ const unsubscribe = store.subscribe(() => {
console.log(store.getState()); console.log(store.getState());
}); });
//Start listening for key events
Keyboard.register();
ReactDOM.render( ReactDOM.render(
( (
<Provider store={store}> <Provider store={store}>

View File

@ -28,17 +28,39 @@ import ButtonGroup from './button/ButtonGroup';
import Form, { FormManager } from './form/Form'; import Form, { FormManager } from './form/Form';
import InputGroup from './group/InputGroup'; import InputGroup from './group/InputGroup';
import Label from './label/Label'; import Label from './label/Label';
import Keyboard from './../keyboard/Keyboard';
export default class Input extends React.Component { export default class Input extends React.Component {
constructor(props) { constructor(props) {
super(props); super(props);
this.state = { this.state = {
value: props.value value: props.value,
focused: false
}; };
} }
isFocused() {
return this.state && this.state.focused === true;
}
componentDidMount() {
if(this.props.manager) this.props.manager.addInput(this);
Keyboard.addListener(this);
}
componentWillUnmount() {
if(this.props.manager) this.props.manager.removeInput(this);
Keyboard.removeListener(this);
}
onKeyUp(e, k) {
if(!this.props.manager) return;
if(!this.isFocused()) return;
if(!k.isSubmit()) return;
this.props.manager.submit();
}
onChange(e, a, b) { onChange(e, a, b) {
//Try my props //Try my props
if(this.props.onChange && this.props.onChange(e) === false) return false; if(this.props.onChange && this.props.onChange(e) === false) return false;
@ -54,12 +76,12 @@ export default class Input extends React.Component {
}); });
} }
componentDidMount() { onFocus() {
if(this.props.manager) this.props.manager.addInput(this); this.setState({ focused: true });
} }
componentWillUnmount() { onBlur() {
if(this.props.manager) this.props.manager.removeInput(this); this.setState({ focused: false });
} }
render() { render() {
@ -119,6 +141,8 @@ export default class Input extends React.Component {
{...this.props} {...this.props}
className={innerClazzes} className={innerClazzes}
onChange={this.onChange.bind(this)} onChange={this.onChange.bind(this)}
onFocus={this.onFocus.bind(this)}
onBlur={this.onBlur.bind(this)}
>{ this.state.value }</textarea> >{ this.state.value }</textarea>
); );
@ -126,6 +150,8 @@ export default class Input extends React.Component {
element = (<ElementType element = (<ElementType
{...this.props} {...this.props}
onChange={this.onChange.bind(this)} onChange={this.onChange.bind(this)}
onFocus={this.onFocus.bind(this)}
onBlur={this.onBlur.bind(this)}
type={type} type={type}
value={ this.state.value } value={ this.state.value }
className={innerClazzes} className={innerClazzes}

View File

@ -57,13 +57,25 @@ export default class Form extends React.Component {
this.state = s; this.state = s;
} }
componentDidMount() {
if(this.props.manager) this.props.manager.addForm(this);
}
componentWillUnmount() {
if(this.props.manager) this.props.manager.removeForm(this);
}
submit() {
this.onSubmit();
}
onSubmit(e) { onSubmit(e) {
//Is Ajax? //Is Ajax?
if(!this.state.ajax) { if(!this.state.ajax) {
return this.state.onSubmit ? this.state.onSubmit(e) : true; return this.state.onSubmit ? this.state.onSubmit(e) : true;
} }
e.preventDefault(); if(e) e.preventDefault();
if(!this.state.action) return console.warning("This form has no action."); if(!this.state.action) return console.warning("This form has no action.");
if(this.state.submitting) return false;//Already submitting? if(this.state.submitting) return false;//Already submitting?
@ -166,6 +178,7 @@ export default class Form extends React.Component {
return ( return (
<form <form
ref="formDOM"
className={clazz} className={clazz}
method={ this.state.method } method={ this.state.method }
autoComplete={ this.props.autoComplete } autoComplete={ this.props.autoComplete }
@ -186,17 +199,27 @@ class FormManager {
this.inputs = []; this.inputs = [];
} }
addInput(input) { addForm(form) { this.forms.push(form); }
this.inputs.push(input); addInput(input) { this.inputs.push(input); }
removeForm(form) {
let i = this.forms.indexOf(form);
if(i === -1) return;
this.forms.splice(i, 1);
} }
removeInput(input) { removeInput(input) {
let i = this.forms.indexOf(input); let i = this.inputs.indexOf(input);
if(i === -1) return; if(i === -1) return;
this.inputs.splice(i, 1); this.inputs.splice(i, 1);
} }
onChange(input, event) { onChange(input, event) {}
submit() {
for(let i = 0; i < this.forms.length; i++) {
this.forms[i].submit();
}
} }
getFormData() { getFormData() {

View File

@ -22,19 +22,77 @@
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
const KEY_ESCAPE = 27; const KEY_ESCAPE = 27;
const KEY_CTRL = 17;
const KEY_ENTER = 13;
class Keyboard { class Keyboard {
constructor(){} constructor(){
//Bound events
this.onKeyUpBound = this.onKeyUp.bind(this);
this.onKeyDownBound = this.onKeyDown.bind(this);
isKeyOrCode(evt, key, code) { this.listeners = [];
if (typeof evt.key !== typeof undefined) {
return evt.key.toLowerCase() === key.toLowerCase(); this.states = {};
}
return evt.keyCode === code;
} }
isEscape(e) { isRegistered() {return this.registered === true;}
return this.isKeyOrCode(e, "Escape", KEY_ESCAPE);
register() {
if(this.isRegistered()) return;
this.registered = true;
document.addEventListener('keyup', this.onKeyUpBound);
document.addEventListener('keydown', this.onKeyDownBound);
}
unregister() {
this.registered = false;
document.removeEventListener('keyup', this.onKeyUpBound);
document.removeEventListener('keydown', this.onKeyDownBound);
}
addListener(listener) {
this.listeners.push(listener);
}
removeListener(listener) {
let index = this.listeners.indexOf(listener);
if(index === -1) return;
this.listeners.splice(index,1);
}
tellListeners(keyCode, event) {
for(let i = 0; i < this.listeners.length; i++) {
let l = this.listeners[i];
if(typeof l[event] !== "function") continue;
l[event](keyCode, this);
}
}
onKeyUp(e) {
//console.log("Key " + e.keyCode); // FOR TESTING KEY COMBOS
this.tellListeners(e.keyCode, "onKeyUp");
this.states[e.keyCode] = false;
this.tellListeners(e.keyCode, "onKeyRelease");
}
onKeyDown(e) {
this.tellListeners(e.keyCode, "onKeyPress");
this.states[e.keyCode] = true;
this.tellListeners(e.keyCode, "onKeyRelease");
}
isKey(key) {
return typeof this.states[key] !== typeof undefined && this.states[key] === true;
}
//Now the submits
isEscape() {
return this.isKey(KEY_ESCAPE);
}
isSubmit() {
return this.isKey(KEY_CTRL) && this.isKey(KEY_ENTER);
} }
} }

View File

@ -34,26 +34,24 @@ import Keyboard from './../keyboard/Keyboard';
class Modal extends React.Component { class Modal extends React.Component {
constructor(props) { constructor(props) {
super(props); super(props);
this.onKeyDownBound = this.onKeyDown.bind(this);
} }
componentDidMount() { componentDidMount() {
document.addEventListener('keydown', this.onKeyDownBound); Keyboard.addListener(this);
} }
componentWillUnmount() { componentWillUnmount() {
document.removeEventListener('keydown', this.onKeyDownBound); Keyboard.removeListener(this);
} }
onKeyDown(e) { onKeyUp(e) {
if(!Keyboard.isEscape(e)) return; if(!Keyboard.isEscape()) return;
e.preventDefault(); e.preventDefault();
e.stopPropagation(); e.stopPropagation();
this.props.closeModal(); this.props.closeModal();
} }
render() { render() {
//Add necessary buttons //Add necessary buttons
let buttons = []; let buttons = [];
if(this.props.buttons) { if(this.props.buttons) {