Declare setting types
This commit is contained in:
		
							parent
							
								
									d01db82c0d
								
							
						
					
					
						commit
						a0882bbceb
					
				
					 48 changed files with 1618 additions and 903 deletions
				
			
		
							
								
								
									
										414
									
								
								src/shared/SettingData.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										414
									
								
								src/shared/SettingData.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,414 @@ | |||
| import * as operations from './operations'; | ||||
| import Settings, * as settings from './Settings'; | ||||
| 
 | ||||
| export class FormKeymaps { | ||||
|   private data: {[op: string]: string}; | ||||
| 
 | ||||
|   constructor(data: {[op: string]: string}) { | ||||
|     this.data = data; | ||||
|   } | ||||
| 
 | ||||
|   toKeymaps(): settings.Keymaps { | ||||
|     let keymaps: settings.Keymaps = {}; | ||||
|     for (let name of Object.keys(this.data)) { | ||||
|       let [type, argStr] = name.split('?'); | ||||
|       let args = {}; | ||||
|       if (argStr) { | ||||
|         args = JSON.parse(argStr); | ||||
|       } | ||||
|       let key = this.data[name]; | ||||
|       keymaps[key] = operations.valueOf({ type, ...args }); | ||||
|     } | ||||
|     return keymaps; | ||||
|   } | ||||
| 
 | ||||
|   toJSON(): {[op: string]: string} { | ||||
|     return this.data; | ||||
|   } | ||||
| 
 | ||||
|   buildWithOverride(op: string, keys: string): FormKeymaps { | ||||
|     let newData = { | ||||
|       ...this.data, | ||||
|       [op]: keys, | ||||
|     }; | ||||
|     return new FormKeymaps(newData); | ||||
|   } | ||||
| 
 | ||||
|   static valueOf(o: ReturnType<FormKeymaps['toJSON']>): FormKeymaps { | ||||
|     let data: {[op: string]: string} = {}; | ||||
|     for (let op of Object.keys(o)) { | ||||
|       data[op] = o[op] as string; | ||||
|     } | ||||
|     return new FormKeymaps(data); | ||||
|   } | ||||
| 
 | ||||
|   static fromKeymaps(keymaps: settings.Keymaps): FormKeymaps { | ||||
|     let data: {[op: string]: string} = {}; | ||||
|     for (let key of Object.keys(keymaps)) { | ||||
|       let op = keymaps[key]; | ||||
|       let args = { ...op }; | ||||
|       delete args.type; | ||||
| 
 | ||||
|       let name = op.type; | ||||
|       if (Object.keys(args).length > 0) { | ||||
|         name += '?' + JSON.stringify(args); | ||||
|       } | ||||
|       data[name] = key; | ||||
|     } | ||||
|     return new FormKeymaps(data); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export class FormSearch { | ||||
|   private default: string; | ||||
| 
 | ||||
|   private engines: string[][]; | ||||
| 
 | ||||
|   constructor(defaultEngine: string, engines: string[][]) { | ||||
|     this.default = defaultEngine; | ||||
|     this.engines = engines; | ||||
|   } | ||||
| 
 | ||||
|   toSearchSettings(): settings.Search { | ||||
|     return { | ||||
|       default: this.default, | ||||
|       engines: this.engines.reduce( | ||||
|         (o: {[key: string]: string}, [name, url]) => { | ||||
|           o[name] = url; | ||||
|           return o; | ||||
|         }, {}), | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   toJSON(): { | ||||
|     default: string; | ||||
|     engines: string[][]; | ||||
|     } { | ||||
|     return { | ||||
|       default: this.default, | ||||
|       engines: this.engines, | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   static valueOf(o: ReturnType<FormSearch['toJSON']>): FormSearch { | ||||
|     if (!Object.prototype.hasOwnProperty.call(o, 'default')) { | ||||
|       throw new TypeError(`"default" field not set`); | ||||
|     } | ||||
|     if (!Object.prototype.hasOwnProperty.call(o, 'engines')) { | ||||
|       throw new TypeError(`"engines" field not set`); | ||||
|     } | ||||
|     return new FormSearch(o.default, o.engines); | ||||
|   } | ||||
| 
 | ||||
|   static fromSearch(search: settings.Search): FormSearch { | ||||
|     let engines = Object.entries(search.engines).reduce( | ||||
|       (o: string[][], [name, url]) => { | ||||
|         return o.concat([[name, url]]); | ||||
|       }, []); | ||||
|     return new FormSearch(search.default, engines); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export class JSONSettings { | ||||
|   private json: string; | ||||
| 
 | ||||
|   constructor(json: any) { | ||||
|     this.json = json; | ||||
|   } | ||||
| 
 | ||||
|   toSettings(): Settings { | ||||
|     return settings.valueOf(JSON.parse(this.json)); | ||||
|   } | ||||
| 
 | ||||
|   toJSON(): string { | ||||
|     return this.json; | ||||
|   } | ||||
| 
 | ||||
|   static valueOf(o: ReturnType<JSONSettings['toJSON']>): JSONSettings { | ||||
|     return new JSONSettings(o); | ||||
|   } | ||||
| 
 | ||||
|   static fromSettings(data: Settings): JSONSettings { | ||||
|     return new JSONSettings(JSON.stringify(data, undefined, 2)); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export class FormSettings { | ||||
|   private keymaps: FormKeymaps; | ||||
| 
 | ||||
|   private search: FormSearch; | ||||
| 
 | ||||
|   private properties: settings.Properties; | ||||
| 
 | ||||
|   private blacklist: string[]; | ||||
| 
 | ||||
|   constructor( | ||||
|     keymaps: FormKeymaps, | ||||
|     search: FormSearch, | ||||
|     properties: settings.Properties, | ||||
|     blacklist: string[], | ||||
|   ) { | ||||
|     this.keymaps = keymaps; | ||||
|     this.search = search; | ||||
|     this.properties = properties; | ||||
|     this.blacklist = blacklist; | ||||
|   } | ||||
| 
 | ||||
|   buildWithKeymaps(keymaps: FormKeymaps): FormSettings { | ||||
|     return new FormSettings( | ||||
|       keymaps, | ||||
|       this.search, | ||||
|       this.properties, | ||||
|       this.blacklist, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   buildWithSearch(search: FormSearch): FormSettings { | ||||
|     return new FormSettings( | ||||
|       this.keymaps, | ||||
|       search, | ||||
|       this.properties, | ||||
|       this.blacklist, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   buildWithProperties(props: settings.Properties): FormSettings { | ||||
|     return new FormSettings( | ||||
|       this.keymaps, | ||||
|       this.search, | ||||
|       props, | ||||
|       this.blacklist, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   buildWithBlacklist(blacklist: string[]): FormSettings { | ||||
|     return new FormSettings( | ||||
|       this.keymaps, | ||||
|       this.search, | ||||
|       this.properties, | ||||
|       blacklist, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   toSettings(): Settings { | ||||
|     return settings.valueOf({ | ||||
|       keymaps: this.keymaps.toKeymaps(), | ||||
|       search: this.search.toSearchSettings(), | ||||
|       properties: this.properties, | ||||
|       blacklist: this.blacklist, | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   toJSON(): { | ||||
|     keymaps: ReturnType<FormKeymaps['toJSON']>; | ||||
|     search: ReturnType<FormSearch['toJSON']>; | ||||
|     properties: settings.Properties; | ||||
|     blacklist: string[]; | ||||
|     } { | ||||
|     return { | ||||
|       keymaps: this.keymaps.toJSON(), | ||||
|       search: this.search.toJSON(), | ||||
|       properties: this.properties, | ||||
|       blacklist: this.blacklist, | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   static valueOf(o: ReturnType<FormSettings['toJSON']>): FormSettings { | ||||
|     for (let name of ['keymaps', 'search', 'properties', 'blacklist']) { | ||||
|       if (!Object.prototype.hasOwnProperty.call(o, name)) { | ||||
|         throw new Error(`"${name}" field not set`); | ||||
|       } | ||||
|     } | ||||
|     return new FormSettings( | ||||
|       FormKeymaps.valueOf(o.keymaps), | ||||
|       FormSearch.valueOf(o.search), | ||||
|       settings.propertiesValueOf(o.properties), | ||||
|       settings.blacklistValueOf(o.blacklist), | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   static fromSettings(data: Settings): FormSettings { | ||||
|     return new FormSettings( | ||||
|       FormKeymaps.fromKeymaps(data.keymaps), | ||||
|       FormSearch.fromSearch(data.search), | ||||
|       data.properties, | ||||
|       data.blacklist); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export enum SettingSource { | ||||
|   JSON = 'json', | ||||
|   Form = 'form', | ||||
| } | ||||
| 
 | ||||
| export default class SettingData { | ||||
|   private source: SettingSource; | ||||
| 
 | ||||
|   private json?: JSONSettings; | ||||
| 
 | ||||
|   private form?: FormSettings; | ||||
| 
 | ||||
|   constructor({ | ||||
|     source, json, form | ||||
|   }: { | ||||
|     source: SettingSource, | ||||
|     json?: JSONSettings, | ||||
|     form?: FormSettings, | ||||
|   }) { | ||||
|     this.source = source; | ||||
|     this.json = json; | ||||
|     this.form = form; | ||||
|   } | ||||
| 
 | ||||
|   getSource(): SettingSource { | ||||
|     return this.source; | ||||
|   } | ||||
| 
 | ||||
|   getJSON(): JSONSettings { | ||||
|     if (!this.json) { | ||||
|       throw new TypeError('json settings not set'); | ||||
|     } | ||||
|     return this.json; | ||||
|   } | ||||
| 
 | ||||
|   getForm(): FormSettings { | ||||
|     if (!this.form) { | ||||
|       throw new TypeError('form settings not set'); | ||||
|     } | ||||
|     return this.form; | ||||
|   } | ||||
| 
 | ||||
|   toJSON(): any { | ||||
|     switch (this.source) { | ||||
|     case SettingSource.JSON: | ||||
|       return { | ||||
|         source: this.source, | ||||
|         json: (this.json as JSONSettings).toJSON(), | ||||
|       }; | ||||
|     case SettingSource.Form: | ||||
|       return { | ||||
|         source: this.source, | ||||
|         form: (this.form as FormSettings).toJSON(), | ||||
|       }; | ||||
|     } | ||||
|     throw new Error(`unknown settings source: ${this.source}`); | ||||
|   } | ||||
| 
 | ||||
|   toSettings(): Settings { | ||||
|     switch (this.source) { | ||||
|     case SettingSource.JSON: | ||||
|       return this.getJSON().toSettings(); | ||||
|     case SettingSource.Form: | ||||
|       return this.getForm().toSettings(); | ||||
|     } | ||||
|     throw new Error(`unknown settings source: ${this.source}`); | ||||
|   } | ||||
| 
 | ||||
|   static valueOf(o: { | ||||
|     source: string; | ||||
|     json?: string; | ||||
|     form?: ReturnType<FormSettings['toJSON']>; | ||||
|   }): SettingData { | ||||
|     switch (o.source) { | ||||
|     case SettingSource.JSON: | ||||
|       return new SettingData({ | ||||
|         source: o.source, | ||||
|         json: JSONSettings.valueOf( | ||||
|           o.json as ReturnType<JSONSettings['toJSON']>), | ||||
|       }); | ||||
|     case SettingSource.Form: | ||||
|       return new SettingData({ | ||||
|         source: o.source, | ||||
|         form: FormSettings.valueOf( | ||||
|           o.form as ReturnType<FormSettings['toJSON']>), | ||||
|       }); | ||||
|     } | ||||
|     throw new Error(`unknown settings source: ${o.source}`); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export const DefaultSettingData: SettingData = SettingData.valueOf({ | ||||
|   source: 'json', | ||||
|   json: `{
 | ||||
|   "keymaps": { | ||||
|     "0": { "type": "scroll.home" }, | ||||
|     ":": { "type": "command.show" }, | ||||
|     "o": { "type": "command.show.open", "alter": false }, | ||||
|     "O": { "type": "command.show.open", "alter": true }, | ||||
|     "t": { "type": "command.show.tabopen", "alter": false }, | ||||
|     "T": { "type": "command.show.tabopen", "alter": true }, | ||||
|     "w": { "type": "command.show.winopen", "alter": false }, | ||||
|     "W": { "type": "command.show.winopen", "alter": true }, | ||||
|     "b": { "type": "command.show.buffer" }, | ||||
|     "a": { "type": "command.show.addbookmark", "alter": true }, | ||||
|     "k": { "type": "scroll.vertically", "count": -1 }, | ||||
|     "j": { "type": "scroll.vertically", "count": 1 }, | ||||
|     "h": { "type": "scroll.horizonally", "count": -1 }, | ||||
|     "l": { "type": "scroll.horizonally", "count": 1 }, | ||||
|     "<C-U>": { "type": "scroll.pages", "count": -0.5 }, | ||||
|     "<C-D>": { "type": "scroll.pages", "count": 0.5 }, | ||||
|     "<C-B>": { "type": "scroll.pages", "count": -1 }, | ||||
|     "<C-F>": { "type": "scroll.pages", "count": 1 }, | ||||
|     "gg": { "type": "scroll.top" }, | ||||
|     "G": { "type": "scroll.bottom" }, | ||||
|     "$": { "type": "scroll.end" }, | ||||
|     "d": { "type": "tabs.close" }, | ||||
|     "D": { "type": "tabs.close.right" }, | ||||
|     "!d": { "type": "tabs.close.force" }, | ||||
|     "u": { "type": "tabs.reopen" }, | ||||
|     "K": { "type": "tabs.prev" }, | ||||
|     "J": { "type": "tabs.next" }, | ||||
|     "gT": { "type": "tabs.prev" }, | ||||
|     "gt": { "type": "tabs.next" }, | ||||
|     "g0": { "type": "tabs.first" }, | ||||
|     "g$": { "type": "tabs.last" }, | ||||
|     "<C-6>": { "type": "tabs.prevsel" }, | ||||
|     "r": { "type": "tabs.reload", "cache": false }, | ||||
|     "R": { "type": "tabs.reload", "cache": true }, | ||||
|     "zp": { "type": "tabs.pin.toggle" }, | ||||
|     "zd": { "type": "tabs.duplicate" }, | ||||
|     "zi": { "type": "zoom.in" }, | ||||
|     "zo": { "type": "zoom.out" }, | ||||
|     "zz": { "type": "zoom.neutral" }, | ||||
|     "f": { "type": "follow.start", "newTab": false }, | ||||
|     "F": { "type": "follow.start", "newTab": true, "background": false }, | ||||
|     "m": { "type": "mark.set.prefix" }, | ||||
|     "'": { "type": "mark.jump.prefix" }, | ||||
|     "H": { "type": "navigate.history.prev" }, | ||||
|     "L": { "type": "navigate.history.next" }, | ||||
|     "[[": { "type": "navigate.link.prev" }, | ||||
|     "]]": { "type": "navigate.link.next" }, | ||||
|     "gu": { "type": "navigate.parent" }, | ||||
|     "gU": { "type": "navigate.root" }, | ||||
|     "gi": { "type": "focus.input" }, | ||||
|     "gf": { "type": "page.source" }, | ||||
|     "gh": { "type": "page.home" }, | ||||
|     "gH": { "type": "page.home", "newTab": true }, | ||||
|     "y": { "type": "urls.yank" }, | ||||
|     "p": { "type": "urls.paste", "newTab": false }, | ||||
|     "P": { "type": "urls.paste", "newTab": true }, | ||||
|     "/": { "type": "find.start" }, | ||||
|     "n": { "type": "find.next" }, | ||||
|     "N": { "type": "find.prev" }, | ||||
|     "<S-Esc>": { "type": "addon.toggle.enabled" } | ||||
|   }, | ||||
|   "search": { | ||||
|     "default": "google", | ||||
|     "engines": { | ||||
|       "google": "https://google.com/search?q={}", | ||||
|       "yahoo": "https://search.yahoo.com/search?p={}", | ||||
|       "bing": "https://www.bing.com/search?q={}", | ||||
|       "duckduckgo": "https://duckduckgo.com/?q={}", | ||||
|       "twitter": "https://twitter.com/search?q={}", | ||||
|       "wikipedia": "https://en.wikipedia.org/w/index.php?search={}" | ||||
|     } | ||||
|   }, | ||||
|   "properties": { | ||||
|     "hintchars": "abcdefghijklmnopqrstuvwxyz", | ||||
|     "smoothscroll": false, | ||||
|     "complete": "sbh" | ||||
|   }, | ||||
|   "blacklist": [ | ||||
|   ] | ||||
| }`,
 | ||||
| }); | ||||
							
								
								
									
										200
									
								
								src/shared/Settings.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										200
									
								
								src/shared/Settings.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,200 @@ | |||
| import * as operations from './operations'; | ||||
| import * as PropertyDefs from './property-defs'; | ||||
| 
 | ||||
| export type Keymaps = {[key: string]: operations.Operation}; | ||||
| 
 | ||||
| export interface Search { | ||||
|   default: string; | ||||
|   engines: { [key: string]: string }; | ||||
| } | ||||
| 
 | ||||
| export interface Properties { | ||||
|   hintchars: string; | ||||
|   smoothscroll: boolean; | ||||
|   complete: string; | ||||
| } | ||||
| 
 | ||||
| export default interface Settings { | ||||
|   keymaps: Keymaps; | ||||
|   search: Search; | ||||
|   properties: Properties; | ||||
|   blacklist: string[]; | ||||
|   // eslint-disable-next-line semi
 | ||||
| } | ||||
| 
 | ||||
| const DefaultProperties: Properties = PropertyDefs.defs.reduce( | ||||
|   (o: {[name: string]: PropertyDefs.Type}, def) => { | ||||
|     o[def.name] = def.defaultValue; | ||||
|     return o; | ||||
|   }, {}) as Properties; | ||||
| 
 | ||||
| 
 | ||||
| export const keymapsValueOf = (o: any): Keymaps => { | ||||
|   return Object.keys(o).reduce((keymaps: Keymaps, key: string): Keymaps => { | ||||
|     let op = operations.valueOf(o[key]); | ||||
|     keymaps[key] = op; | ||||
|     return keymaps; | ||||
|   }, {}); | ||||
| }; | ||||
| 
 | ||||
| export const searchValueOf = (o: any): Search => { | ||||
|   if (typeof o.default !== 'string') { | ||||
|     throw new TypeError('string field "default" not set"'); | ||||
|   } | ||||
|   for (let name of Object.keys(o.engines)) { | ||||
|     if ((/\s/).test(name)) { | ||||
|       throw new TypeError( | ||||
|         `While space in the search engine not allowed: "${name}"`); | ||||
|     } | ||||
|     let url = o.engines[name]; | ||||
|     if (typeof url !== 'string') { | ||||
|       throw new TypeError('"engines" not an object of string'); | ||||
|     } | ||||
|     let matches = url.match(/{}/g); | ||||
|     if (matches === null) { | ||||
|       throw new TypeError(`No {}-placeholders in URL of "${name}"`); | ||||
|     } else if (matches.length > 1) { | ||||
|       throw new TypeError(`Multiple {}-placeholders in URL of "${name}"`); | ||||
|     } | ||||
| 
 | ||||
|   } | ||||
|   if (!Object.prototype.hasOwnProperty.call(o.engines, o.default)) { | ||||
|     throw new TypeError(`Default engine "${o.default}" not found`); | ||||
|   } | ||||
|   return { | ||||
|     default: o.default as string, | ||||
|     engines: { ...o.engines }, | ||||
|   }; | ||||
| }; | ||||
| 
 | ||||
| export const propertiesValueOf = (o: any): Properties => { | ||||
|   let defNames = new Set(PropertyDefs.defs.map(def => def.name)); | ||||
|   let unknownName = Object.keys(o).find(name => !defNames.has(name)); | ||||
|   if (unknownName) { | ||||
|     throw new TypeError(`Unknown property name: "${unknownName}"`); | ||||
|   } | ||||
| 
 | ||||
|   for (let def of PropertyDefs.defs) { | ||||
|     if (!Object.prototype.hasOwnProperty.call(o, def.name)) { | ||||
|       continue; | ||||
|     } | ||||
|     if (typeof o[def.name] !== def.type) { | ||||
|       throw new TypeError(`property "${def.name}" is not ${def.type}`); | ||||
|     } | ||||
|   } | ||||
|   return { | ||||
|     ...DefaultProperties, | ||||
|     ...o, | ||||
|   }; | ||||
| }; | ||||
| 
 | ||||
| export const blacklistValueOf = (o: any): string[] => { | ||||
|   if (!Array.isArray(o)) { | ||||
|     throw new TypeError(`"blacklist" is not an array of string`); | ||||
|   } | ||||
|   for (let x of o) { | ||||
|     if (typeof x !== 'string') { | ||||
|       throw new TypeError(`"blacklist" is not an array of string`); | ||||
|     } | ||||
|   } | ||||
|   return o as string[]; | ||||
| }; | ||||
| 
 | ||||
| export const valueOf = (o: any): Settings => { | ||||
|   let settings = { ...DefaultSetting }; | ||||
|   if (Object.prototype.hasOwnProperty.call(o, 'keymaps')) { | ||||
|     settings.keymaps = keymapsValueOf(o.keymaps); | ||||
|   } | ||||
|   if (Object.prototype.hasOwnProperty.call(o, 'search')) { | ||||
|     settings.search = searchValueOf(o.search); | ||||
|   } | ||||
|   if (Object.prototype.hasOwnProperty.call(o, 'properties')) { | ||||
|     settings.properties = propertiesValueOf(o.properties); | ||||
|   } | ||||
|   if (Object.prototype.hasOwnProperty.call(o, 'blacklist')) { | ||||
|     settings.blacklist = blacklistValueOf(o.blacklist); | ||||
|   } | ||||
|   return settings; | ||||
| }; | ||||
| 
 | ||||
| const DefaultSetting: Settings = { | ||||
|   keymaps: { | ||||
|     '0': { 'type': 'scroll.home' }, | ||||
|     ':': { 'type': 'command.show' }, | ||||
|     'o': { 'type': 'command.show.open', 'alter': false }, | ||||
|     'O': { 'type': 'command.show.open', 'alter': true }, | ||||
|     't': { 'type': 'command.show.tabopen', 'alter': false }, | ||||
|     'T': { 'type': 'command.show.tabopen', 'alter': true }, | ||||
|     'w': { 'type': 'command.show.winopen', 'alter': false }, | ||||
|     'W': { 'type': 'command.show.winopen', 'alter': true }, | ||||
|     'b': { 'type': 'command.show.buffer' }, | ||||
|     'a': { 'type': 'command.show.addbookmark', 'alter': true }, | ||||
|     'k': { 'type': 'scroll.vertically', 'count': -1 }, | ||||
|     'j': { 'type': 'scroll.vertically', 'count': 1 }, | ||||
|     'h': { 'type': 'scroll.horizonally', 'count': -1 }, | ||||
|     'l': { 'type': 'scroll.horizonally', 'count': 1 }, | ||||
|     '<C-U>': { 'type': 'scroll.pages', 'count': -0.5 }, | ||||
|     '<C-D>': { 'type': 'scroll.pages', 'count': 0.5 }, | ||||
|     '<C-B>': { 'type': 'scroll.pages', 'count': -1 }, | ||||
|     '<C-F>': { 'type': 'scroll.pages', 'count': 1 }, | ||||
|     'gg': { 'type': 'scroll.top' }, | ||||
|     'G': { 'type': 'scroll.bottom' }, | ||||
|     '$': { 'type': 'scroll.end' }, | ||||
|     'd': { 'type': 'tabs.close' }, | ||||
|     'D': { 'type': 'tabs.close.right' }, | ||||
|     '!d': { 'type': 'tabs.close.force' }, | ||||
|     'u': { 'type': 'tabs.reopen' }, | ||||
|     'K': { 'type': 'tabs.prev' }, | ||||
|     'J': { 'type': 'tabs.next' }, | ||||
|     'gT': { 'type': 'tabs.prev' }, | ||||
|     'gt': { 'type': 'tabs.next' }, | ||||
|     'g0': { 'type': 'tabs.first' }, | ||||
|     'g$': { 'type': 'tabs.last' }, | ||||
|     '<C-6>': { 'type': 'tabs.prevsel' }, | ||||
|     'r': { 'type': 'tabs.reload', 'cache': false }, | ||||
|     'R': { 'type': 'tabs.reload', 'cache': true }, | ||||
|     'zp': { 'type': 'tabs.pin.toggle' }, | ||||
|     'zd': { 'type': 'tabs.duplicate' }, | ||||
|     'zi': { 'type': 'zoom.in' }, | ||||
|     'zo': { 'type': 'zoom.out' }, | ||||
|     'zz': { 'type': 'zoom.neutral' }, | ||||
|     'f': { 'type': 'follow.start', 'newTab': false, 'background': false }, | ||||
|     'F': { 'type': 'follow.start', 'newTab': true, 'background': false }, | ||||
|     'm': { 'type': 'mark.set.prefix' }, | ||||
|     '\'': { 'type': 'mark.jump.prefix' }, | ||||
|     'H': { 'type': 'navigate.history.prev' }, | ||||
|     'L': { 'type': 'navigate.history.next' }, | ||||
|     '[[': { 'type': 'navigate.link.prev' }, | ||||
|     ']]': { 'type': 'navigate.link.next' }, | ||||
|     'gu': { 'type': 'navigate.parent' }, | ||||
|     'gU': { 'type': 'navigate.root' }, | ||||
|     'gi': { 'type': 'focus.input' }, | ||||
|     'gf': { 'type': 'page.source' }, | ||||
|     'gh': { 'type': 'page.home', 'newTab': false }, | ||||
|     'gH': { 'type': 'page.home', 'newTab': true }, | ||||
|     'y': { 'type': 'urls.yank' }, | ||||
|     'p': { 'type': 'urls.paste', 'newTab': false }, | ||||
|     'P': { 'type': 'urls.paste', 'newTab': true }, | ||||
|     '/': { 'type': 'find.start' }, | ||||
|     'n': { 'type': 'find.next' }, | ||||
|     'N': { 'type': 'find.prev' }, | ||||
|     '<S-Esc>': { 'type': 'addon.toggle.enabled' } | ||||
|   }, | ||||
|   search: { | ||||
|     default: 'google', | ||||
|     engines: { | ||||
|       'google': 'https://google.com/search?q={}', | ||||
|       'yahoo': 'https://search.yahoo.com/search?p={}', | ||||
|       'bing': 'https://www.bing.com/search?q={}', | ||||
|       'duckduckgo': 'https://duckduckgo.com/?q={}', | ||||
|       'twitter': 'https://twitter.com/search?q={}', | ||||
|       'wikipedia': 'https://en.wikipedia.org/w/index.php?search={}' | ||||
|     } | ||||
|   }, | ||||
|   properties: { | ||||
|     hintchars: 'abcdefghijklmnopqrstuvwxyz', | ||||
|     smoothscroll: false, | ||||
|     complete: 'sbh' | ||||
|   }, | ||||
|   blacklist: [] | ||||
| }; | ||||
|  | @ -443,5 +443,5 @@ export const valueOf = (o: any): Operation => { | |||
|   case MARK_JUMP_PREFIX: | ||||
|     return { type: o.type }; | ||||
|   } | ||||
|   throw new Error('unknown operation type: ' + o.type); | ||||
|   throw new TypeError('unknown operation type: ' + o.type); | ||||
| }; | ||||
|  |  | |||
							
								
								
									
										50
									
								
								src/shared/properties.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/shared/properties.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,50 @@ | |||
| export type Type = string | number | boolean; | ||||
| 
 | ||||
| export class Def { | ||||
|   private name0: string; | ||||
| 
 | ||||
|   private description0: string; | ||||
| 
 | ||||
|   private defaultValue0: Type; | ||||
| 
 | ||||
|   constructor( | ||||
|     name: string, | ||||
|     description: string, | ||||
|     defaultValue: Type, | ||||
|   ) { | ||||
|     this.name0 = name; | ||||
|     this.description0 = description; | ||||
|     this.defaultValue0 = defaultValue; | ||||
|   } | ||||
| 
 | ||||
|   public get name(): string { | ||||
|     return this.name0; | ||||
|   } | ||||
| 
 | ||||
|   public get defaultValue(): Type { | ||||
|     return this.defaultValue0; | ||||
|   } | ||||
| 
 | ||||
|   public get description(): Type { | ||||
|     return this.description0; | ||||
|   } | ||||
| 
 | ||||
|   public get type(): string { | ||||
|     return typeof this.defaultValue; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export const defs: Def[] = [ | ||||
|   new Def( | ||||
|     'hintchars', | ||||
|     'hint characters on follow mode', | ||||
|     'abcdefghijklmnopqrstuvwxyz'), | ||||
|   new Def( | ||||
|     'smoothscroll', | ||||
|     'smooth scroll', | ||||
|     false), | ||||
|   new Def( | ||||
|     'complete', | ||||
|     'which are completed at the open page', | ||||
|     'sbh'), | ||||
| ]; | ||||
							
								
								
									
										50
									
								
								src/shared/property-defs.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/shared/property-defs.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,50 @@ | |||
| export type Type = string | number | boolean; | ||||
| 
 | ||||
| export class Def { | ||||
|   private name0: string; | ||||
| 
 | ||||
|   private description0: string; | ||||
| 
 | ||||
|   private defaultValue0: Type; | ||||
| 
 | ||||
|   constructor( | ||||
|     name: string, | ||||
|     description: string, | ||||
|     defaultValue: Type, | ||||
|   ) { | ||||
|     this.name0 = name; | ||||
|     this.description0 = description; | ||||
|     this.defaultValue0 = defaultValue; | ||||
|   } | ||||
| 
 | ||||
|   public get name(): string { | ||||
|     return this.name0; | ||||
|   } | ||||
| 
 | ||||
|   public get defaultValue(): Type { | ||||
|     return this.defaultValue0; | ||||
|   } | ||||
| 
 | ||||
|   public get description(): Type { | ||||
|     return this.description0; | ||||
|   } | ||||
| 
 | ||||
|   public get type(): string { | ||||
|     return typeof this.defaultValue; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export const defs: Def[] = [ | ||||
|   new Def( | ||||
|     'hintchars', | ||||
|     'hint characters on follow mode', | ||||
|     'abcdefghijklmnopqrstuvwxyz'), | ||||
|   new Def( | ||||
|     'smoothscroll', | ||||
|     'smooth scroll', | ||||
|     false), | ||||
|   new Def( | ||||
|     'complete', | ||||
|     'which are completed at the open page', | ||||
|     'sbh'), | ||||
| ]; | ||||
|  | @ -1,85 +0,0 @@ | |||
| export default { | ||||
|   source: 'json', | ||||
|   json: `{
 | ||||
|   "keymaps": { | ||||
|     "0": { "type": "scroll.home" }, | ||||
|     ":": { "type": "command.show" }, | ||||
|     "o": { "type": "command.show.open", "alter": false }, | ||||
|     "O": { "type": "command.show.open", "alter": true }, | ||||
|     "t": { "type": "command.show.tabopen", "alter": false }, | ||||
|     "T": { "type": "command.show.tabopen", "alter": true }, | ||||
|     "w": { "type": "command.show.winopen", "alter": false }, | ||||
|     "W": { "type": "command.show.winopen", "alter": true }, | ||||
|     "b": { "type": "command.show.buffer" }, | ||||
|     "a": { "type": "command.show.addbookmark", "alter": true }, | ||||
|     "k": { "type": "scroll.vertically", "count": -1 }, | ||||
|     "j": { "type": "scroll.vertically", "count": 1 }, | ||||
|     "h": { "type": "scroll.horizonally", "count": -1 }, | ||||
|     "l": { "type": "scroll.horizonally", "count": 1 }, | ||||
|     "<C-U>": { "type": "scroll.pages", "count": -0.5 }, | ||||
|     "<C-D>": { "type": "scroll.pages", "count": 0.5 }, | ||||
|     "<C-B>": { "type": "scroll.pages", "count": -1 }, | ||||
|     "<C-F>": { "type": "scroll.pages", "count": 1 }, | ||||
|     "gg": { "type": "scroll.top" }, | ||||
|     "G": { "type": "scroll.bottom" }, | ||||
|     "$": { "type": "scroll.end" }, | ||||
|     "d": { "type": "tabs.close" }, | ||||
|     "D": { "type": "tabs.close.right" }, | ||||
|     "!d": { "type": "tabs.close.force" }, | ||||
|     "u": { "type": "tabs.reopen" }, | ||||
|     "K": { "type": "tabs.prev", "count": 1 }, | ||||
|     "J": { "type": "tabs.next", "count": 1 }, | ||||
|     "gT": { "type": "tabs.prev", "count": 1 }, | ||||
|     "gt": { "type": "tabs.next", "count": 1 }, | ||||
|     "g0": { "type": "tabs.first" }, | ||||
|     "g$": { "type": "tabs.last" }, | ||||
|     "<C-6>": { "type": "tabs.prevsel" }, | ||||
|     "r": { "type": "tabs.reload", "cache": false }, | ||||
|     "R": { "type": "tabs.reload", "cache": true }, | ||||
|     "zp": { "type": "tabs.pin.toggle" }, | ||||
|     "zd": { "type": "tabs.duplicate" }, | ||||
|     "zi": { "type": "zoom.in" }, | ||||
|     "zo": { "type": "zoom.out" }, | ||||
|     "zz": { "type": "zoom.neutral" }, | ||||
|     "f": { "type": "follow.start", "newTab": false }, | ||||
|     "F": { "type": "follow.start", "newTab": true, "background": false }, | ||||
|     "m": { "type": "mark.set.prefix" }, | ||||
|     "'": { "type": "mark.jump.prefix" }, | ||||
|     "H": { "type": "navigate.history.prev" }, | ||||
|     "L": { "type": "navigate.history.next" }, | ||||
|     "[[": { "type": "navigate.link.prev" }, | ||||
|     "]]": { "type": "navigate.link.next" }, | ||||
|     "gu": { "type": "navigate.parent" }, | ||||
|     "gU": { "type": "navigate.root" }, | ||||
|     "gi": { "type": "focus.input" }, | ||||
|     "gf": { "type": "page.source" }, | ||||
|     "gh": { "type": "page.home" }, | ||||
|     "gH": { "type": "page.home", "newTab": true }, | ||||
|     "y": { "type": "urls.yank" }, | ||||
|     "p": { "type": "urls.paste", "newTab": false }, | ||||
|     "P": { "type": "urls.paste", "newTab": true }, | ||||
|     "/": { "type": "find.start" }, | ||||
|     "n": { "type": "find.next" }, | ||||
|     "N": { "type": "find.prev" }, | ||||
|     "<S-Esc>": { "type": "addon.toggle.enabled" } | ||||
|   }, | ||||
|   "search": { | ||||
|     "default": "google", | ||||
|     "engines": { | ||||
|       "google": "https://google.com/search?q={}", | ||||
|       "yahoo": "https://search.yahoo.com/search?p={}", | ||||
|       "bing": "https://www.bing.com/search?q={}", | ||||
|       "duckduckgo": "https://duckduckgo.com/?q={}", | ||||
|       "twitter": "https://twitter.com/search?q={}", | ||||
|       "wikipedia": "https://en.wikipedia.org/w/index.php?search={}" | ||||
|     } | ||||
|   }, | ||||
|   "properties": { | ||||
|     "hintchars": "abcdefghijklmnopqrstuvwxyz", | ||||
|     "smoothscroll": false, | ||||
|     "complete": "sbh" | ||||
|   }, | ||||
|   "blacklist": [ | ||||
|   ] | ||||
| }`,
 | ||||
| }; | ||||
|  | @ -1,24 +0,0 @@ | |||
| // describe types of a propety as:
 | ||||
| //    mystr: 'string',
 | ||||
| //    mynum: 'number',
 | ||||
| //    mybool: 'boolean',
 | ||||
| const types: { [key: string]: string } = { | ||||
|   hintchars: 'string', | ||||
|   smoothscroll: 'boolean', | ||||
|   complete: 'string', | ||||
| }; | ||||
| 
 | ||||
| // describe default values of a property
 | ||||
| const defaults: { [key: string]: string | number | boolean } = { | ||||
|   hintchars: 'abcdefghijklmnopqrstuvwxyz', | ||||
|   smoothscroll: false, | ||||
|   complete: 'sbh', | ||||
| }; | ||||
| 
 | ||||
| const docs: { [key: string]: string } = { | ||||
|   hintchars: 'hint characters on follow mode', | ||||
|   smoothscroll: 'smooth scroll', | ||||
|   complete: 'which are completed at the open page', | ||||
| }; | ||||
| 
 | ||||
| export { types, defaults, docs }; | ||||
|  | @ -1,32 +0,0 @@ | |||
| import DefaultSettings from './default'; | ||||
| import * as settingsValues from './values'; | ||||
| 
 | ||||
| const loadRaw = async(): Promise<any> => { | ||||
|   let { settings } = await browser.storage.local.get('settings'); | ||||
|   if (!settings) { | ||||
|     return DefaultSettings; | ||||
|   } | ||||
|   return { ...DefaultSettings, ...settings as object }; | ||||
| }; | ||||
| 
 | ||||
| const loadValue = async() => { | ||||
|   let settings = await loadRaw(); | ||||
|   let value = JSON.parse(DefaultSettings.json); | ||||
|   if (settings.source === 'json') { | ||||
|     value = settingsValues.valueFromJson(settings.json); | ||||
|   } else if (settings.source === 'form') { | ||||
|     value = settingsValues.valueFromForm(settings.form); | ||||
|   } | ||||
|   if (!value.properties) { | ||||
|     value.properties = {}; | ||||
|   } | ||||
|   return { ...settingsValues.valueFromJson(DefaultSettings.json), ...value }; | ||||
| }; | ||||
| 
 | ||||
| const save = (settings: any): Promise<any> => { | ||||
|   return browser.storage.local.set({ | ||||
|     settings, | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
| export { loadRaw, loadValue, save }; | ||||
|  | @ -1,76 +0,0 @@ | |||
| import * as operations from '../operations'; | ||||
| import * as properties from './properties'; | ||||
| 
 | ||||
| const VALID_TOP_KEYS = ['keymaps', 'search', 'blacklist', 'properties']; | ||||
| const VALID_OPERATION_VALUES = Object.keys(operations).map((key) => { | ||||
|   return operations[key]; | ||||
| }); | ||||
| 
 | ||||
| const validateInvalidTopKeys = (settings: any): void => { | ||||
|   let invalidKey = Object.keys(settings).find((key) => { | ||||
|     return !VALID_TOP_KEYS.includes(key); | ||||
|   }); | ||||
|   if (invalidKey) { | ||||
|     throw Error(`Unknown key: "${invalidKey}"`); | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| const validateKeymaps = (keymaps: any): void => { | ||||
|   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 validateSearch = (search: any): void => { | ||||
|   let engines = search.engines; | ||||
|   for (let key of Object.keys(engines)) { | ||||
|     if ((/\s/).test(key)) { | ||||
|       throw new Error( | ||||
|         `While space in search engine name is not allowed: "${key}"` | ||||
|       ); | ||||
|     } | ||||
|     let url = engines[key]; | ||||
|     if (!url.match(/{}/)) { | ||||
|       throw new Error(`No {}-placeholders in URL of "${key}"`); | ||||
|     } | ||||
|     if (url.match(/{}/g).length > 1) { | ||||
|       throw new Error(`Multiple {}-placeholders in URL of "${key}"`); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   if (!search.default) { | ||||
|     throw new Error(`Default engine is not set`); | ||||
|   } | ||||
|   if (!Object.keys(engines).includes(search.default)) { | ||||
|     throw new Error(`Default engine "${search.default}" not found`); | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| const validateProperties = (props: any): void => { | ||||
|   for (let name of Object.keys(props)) { | ||||
|     if (!properties.types[name]) { | ||||
|       throw new Error(`Unknown property name: "${name}"`); | ||||
|     } | ||||
|     if (typeof props[name] !== properties.types[name]) { | ||||
|       throw new Error(`Invalid type for property: "${name}"`); | ||||
|     } | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| const validate = (settings: any): void => { | ||||
|   validateInvalidTopKeys(settings); | ||||
|   if (settings.keymaps) { | ||||
|     validateKeymaps(settings.keymaps); | ||||
|   } | ||||
|   if (settings.search) { | ||||
|     validateSearch(settings.search); | ||||
|   } | ||||
|   if (settings.properties) { | ||||
|     validateProperties(settings.properties); | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| export { validate }; | ||||
|  | @ -1,108 +0,0 @@ | |||
| import * as properties from './properties'; | ||||
| 
 | ||||
| const operationFromFormName = (name: string): any => { | ||||
|   let [type, argStr] = name.split('?'); | ||||
|   let args = {}; | ||||
|   if (argStr) { | ||||
|     args = JSON.parse(argStr); | ||||
|   } | ||||
|   return { type, ...args }; | ||||
| }; | ||||
| 
 | ||||
| const operationToFormName = (op: any): string => { | ||||
|   let type = op.type; | ||||
|   let args = { ...op }; | ||||
|   delete args.type; | ||||
| 
 | ||||
|   if (Object.keys(args).length === 0) { | ||||
|     return type; | ||||
|   } | ||||
|   return op.type + '?' + JSON.stringify(args); | ||||
| }; | ||||
| 
 | ||||
| const valueFromJson = (json: string): object => { | ||||
|   return JSON.parse(json); | ||||
| }; | ||||
| 
 | ||||
| const valueFromForm = (form: any): object => { | ||||
|   let keymaps: any = undefined; | ||||
|   if (form.keymaps) { | ||||
|     keymaps = {}; | ||||
|     for (let name of Object.keys(form.keymaps)) { | ||||
|       let keys = form.keymaps[name]; | ||||
|       keymaps[keys] = operationFromFormName(name); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   let search: any = undefined; | ||||
|   if (form.search) { | ||||
|     search = { default: form.search.default }; | ||||
| 
 | ||||
|     if (form.search.engines) { | ||||
|       search.engines = {}; | ||||
|       for (let [name, url] of form.search.engines) { | ||||
|         search.engines[name] = url; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return { | ||||
|     keymaps, | ||||
|     search, | ||||
|     blacklist: form.blacklist, | ||||
|     properties: form.properties | ||||
|   }; | ||||
| }; | ||||
| 
 | ||||
| const jsonFromValue = (value: any): string => { | ||||
|   return JSON.stringify(value, undefined, 2); | ||||
| }; | ||||
| 
 | ||||
| const formFromValue = (value: any, allowedOps: any[]): any => { | ||||
|   let keymaps: any = undefined; | ||||
| 
 | ||||
|   if (value.keymaps) { | ||||
|     let allowedSet = new Set(allowedOps); | ||||
| 
 | ||||
|     keymaps = {}; | ||||
|     for (let keys of Object.keys(value.keymaps)) { | ||||
|       let op = operationToFormName(value.keymaps[keys]); | ||||
|       if (allowedSet.has(op)) { | ||||
|         keymaps[op] = keys; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   let search: any = undefined; | ||||
|   if (value.search) { | ||||
|     search = { default: value.search.default }; | ||||
|     if (value.search.engines) { | ||||
|       search.engines = Object.keys(value.search.engines).map((name) => { | ||||
|         return [name, value.search.engines[name]]; | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   let formProperties = { ...properties.defaults, ...value.properties }; | ||||
| 
 | ||||
|   return { | ||||
|     keymaps, | ||||
|     search, | ||||
|     blacklist: value.blacklist, | ||||
|     properties: formProperties, | ||||
|   }; | ||||
| }; | ||||
| 
 | ||||
| const jsonFromForm = (form: any): string => { | ||||
|   return jsonFromValue(valueFromForm(form)); | ||||
| }; | ||||
| 
 | ||||
| const formFromJson = (json: string, allowedOps: any[]): any => { | ||||
|   let value = valueFromJson(json); | ||||
|   return formFromValue(value, allowedOps); | ||||
| }; | ||||
| 
 | ||||
| export { | ||||
|   valueFromJson, valueFromForm, jsonFromValue, formFromValue, | ||||
|   jsonFromForm, formFromJson | ||||
| }; | ||||
		Reference in a new issue