aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorShin'ya Ueoka <ueokande@i-beam.org>2017-10-01 12:25:49 +0900
committerShin'ya Ueoka <ueokande@i-beam.org>2017-10-01 12:25:49 +0900
commit1b2554adee528da1078ad39694a3f8264c639821 (patch)
tree0fcacdad33a51cf7611af77e601443eb827c2498 /src
parent709fa95aa383b75e88fb12512f772ea55d757b06 (diff)
parent8c9051076f7782b898ab2f848b9afe4a694c3065 (diff)
Merge branch 'settings-validation'
Diffstat (limited to 'src')
-rw-r--r--src/actions/index.js5
-rw-r--r--src/actions/setting.js27
-rw-r--r--src/components/setting.js45
-rw-r--r--src/pages/settings.js29
-rw-r--r--src/reducers/setting.js17
-rw-r--r--src/shared/validators/setting.js33
6 files changed, 137 insertions, 19 deletions
diff --git a/src/actions/index.js b/src/actions/index.js
index 2aa28fa..63c36d2 100644
--- a/src/actions/index.js
+++ b/src/actions/index.js
@@ -13,5 +13,8 @@ export default {
// Completion
COMPLETION_SET_ITEMS: 'completion.set.items',
COMPLETION_SELECT_NEXT: 'completions.select.next',
- COMPLETION_SELECT_PREV: 'completions.select.prev'
+ COMPLETION_SELECT_PREV: 'completions.select.prev',
+
+ // Settings
+ SETTING_SET_SETTINGS: 'setting.set.settings',
};
diff --git a/src/actions/setting.js b/src/actions/setting.js
new file mode 100644
index 0000000..d8e889e
--- /dev/null
+++ b/src/actions/setting.js
@@ -0,0 +1,27 @@
+import actions from '../actions';
+import messages from '../content/messages';
+
+const load = () => {
+ return browser.storage.local.get('settings').then((value) => {
+ return set(value.settings);
+ }, console.error);
+};
+
+const save = (settings) => {
+ return browser.storage.local.set({
+ settings
+ }).then(() => {
+ return browser.runtime.sendMessage({
+ type: messages.SETTINGS_RELOAD
+ });
+ });
+};
+
+const set = (settings) => {
+ return {
+ type: actions.SETTING_SET_SETTINGS,
+ settings,
+ };
+};
+
+export { load, save, set };
diff --git a/src/components/setting.js b/src/components/setting.js
new file mode 100644
index 0000000..1f3b3fe
--- /dev/null
+++ b/src/components/setting.js
@@ -0,0 +1,45 @@
+import * as settingActions from '../actions/setting';
+import { validate } from '../shared/validators/setting';
+
+export default class SettingComponent {
+ constructor(wrapper, store) {
+ this.wrapper = wrapper;
+ this.store = store;
+
+ let doc = wrapper.ownerDocument;
+ let form = doc.getElementById('vimvixen-settings-form');
+ form.addEventListener('submit', this.onSubmit.bind(this));
+
+ let plainJson = form.elements['plain-json'];
+ plainJson.addEventListener('input', this.onPlainJsonChanged.bind(this));
+
+ store.dispatch(settingActions.load());
+ }
+
+ onSubmit(e) {
+ let settings = {
+ json: e.target.elements['plain-json'].value,
+ };
+ this.store.dispatch(settingActions.save(settings));
+ e.preventDefault();
+ }
+
+ onPlainJsonChanged(e) {
+ try {
+ let settings = JSON.parse(e.target.value);
+ validate(settings);
+ e.target.setCustomValidity('');
+ } catch (err) {
+ e.target.setCustomValidity(err.message);
+ }
+ }
+
+ update() {
+ let { settings } = this.store.getState();
+
+ let doc = this.wrapper.ownerDocument;
+ let form = doc.getElementById('vimvixen-settings-form');
+ let plainJsonInput = form.elements['plain-json'];
+ plainJsonInput.value = settings.json;
+ }
+}
diff --git a/src/pages/settings.js b/src/pages/settings.js
index 6e00ed3..9bad967 100644
--- a/src/pages/settings.js
+++ b/src/pages/settings.js
@@ -1,22 +1,15 @@
import './settings.scss';
-import messages from '../content/messages';
+import SettingComponent from '../components/setting';
+import settingReducer from '../reducers/setting';
+import * as store from '../store';
-document.addEventListener('DOMContentLoaded', () => {
- let form = document.getElementById('vimvixen-settings-form');
- form.addEventListener('submit', (e) => {
- e.preventDefault();
- browser.storage.local.set({
- settings: {
- json: e.target.elements['plain-json'].value
- }
- }).then(() => {
- return browser.runtime.sendMessage({
- type: messages.SETTINGS_RELOAD
- });
- });
- });
+const settingStore = store.createStore(settingReducer);
+let settingComponent = null;
+
+settingStore.subscribe(() => {
+ settingComponent.update();
+});
- browser.storage.local.get('settings').then((value) => {
- form.elements['plain-json'].value = value.settings.json;
- }, console.error);
+document.addEventListener('DOMContentLoaded', () => {
+ settingComponent = new SettingComponent(document.body, settingStore);
});
diff --git a/src/reducers/setting.js b/src/reducers/setting.js
new file mode 100644
index 0000000..735d4fb
--- /dev/null
+++ b/src/reducers/setting.js
@@ -0,0 +1,17 @@
+import actions from '../actions';
+
+const defaultState = {
+ settings: {}
+};
+
+export default function reducer(state = defaultState, action = {}) {
+ switch (action.type) {
+ case actions.SETTING_SET_SETTINGS:
+ return Object.assign({}, state, {
+ settings: action.settings,
+ });
+ default:
+ return state;
+ }
+}
+
diff --git a/src/shared/validators/setting.js b/src/shared/validators/setting.js
new file mode 100644
index 0000000..df04e50
--- /dev/null
+++ b/src/shared/validators/setting.js
@@ -0,0 +1,33 @@
+import operations from '../../operations';
+
+const VALID_TOP_KEYS = ['keymaps'];
+const VALID_OPERATION_VALUES = Object.keys(operations).map((key) => {
+ return operations[key];
+});
+
+const validateInvalidTopKeys = (settings) => {
+ let invalidKey = Object.keys(settings).find((key) => {
+ return !VALID_TOP_KEYS.includes(key);
+ });
+ if (invalidKey) {
+ throw Error(`Unknown key: "${invalidKey}"`);
+ }
+};
+
+const validateKeymaps = (keymaps) => {
+ for (let key of Object.keys(keymaps)) {
+ let value = keymaps[key];
+ if (!VALID_OPERATION_VALUES.includes(value.type)) {
+ throw Error(`Unknown operation: "${value.type}"`);
+ }
+ }
+};
+
+const validate = (settings) => {
+ validateInvalidTopKeys(settings);
+ if (settings.keymaps) {
+ validateKeymaps(settings.keymaps);
+ }
+};
+
+export { validate };