mirror of
https://github.com/Eugeny/tabby.git
synced 2025-06-23 12:59:54 +00:00
284 lines
8.0 KiB
TypeScript
284 lines
8.0 KiB
TypeScript
import { Injectable, Inject, NgZone, EventEmitter } from '@angular/core'
|
|
import { IHotkeyDescription, HotkeyProvider } from '../api/hotkeyProvider'
|
|
import { NativeKeyEvent, stringifyKeySequence } from './hotkeys.util'
|
|
import { ConfigService } from '../services/config.service'
|
|
import { ElectronService } from '../services/electron.service'
|
|
|
|
export interface PartialHotkeyMatch {
|
|
id: string,
|
|
strokes: string[],
|
|
matchedLength: number,
|
|
}
|
|
|
|
const KEY_TIMEOUT = 2000
|
|
|
|
interface EventBufferEntry {
|
|
event: NativeKeyEvent,
|
|
time: number,
|
|
}
|
|
|
|
@Injectable({ providedIn: 'root' })
|
|
export class HotkeysService {
|
|
key = new EventEmitter<NativeKeyEvent>()
|
|
matchedHotkey = new EventEmitter<string>()
|
|
globalHotkey = new EventEmitter()
|
|
private currentKeystrokes: EventBufferEntry[] = []
|
|
private disabledLevel = 0
|
|
private hotkeyDescriptions: IHotkeyDescription[] = []
|
|
|
|
constructor (
|
|
private zone: NgZone,
|
|
private electron: ElectronService,
|
|
private config: ConfigService,
|
|
@Inject(HotkeyProvider) private hotkeyProviders: HotkeyProvider[],
|
|
) {
|
|
let events = ['keydown', 'keyup']
|
|
events.forEach((event) => {
|
|
document.addEventListener(event, (nativeEvent) => {
|
|
if (document.querySelectorAll('input:focus').length === 0) {
|
|
this.pushKeystroke(event, nativeEvent)
|
|
this.processKeystrokes()
|
|
this.emitKeyEvent(nativeEvent)
|
|
}
|
|
})
|
|
})
|
|
this.config.changed$.subscribe(() => {
|
|
this.registerGlobalHotkey()
|
|
})
|
|
this.registerGlobalHotkey()
|
|
this.getHotkeyDescriptions().then(hotkeys => {
|
|
this.hotkeyDescriptions = hotkeys
|
|
})
|
|
}
|
|
|
|
pushKeystroke (name, nativeEvent) {
|
|
nativeEvent.event = name
|
|
this.currentKeystrokes.push({ event: nativeEvent, time: performance.now() })
|
|
}
|
|
|
|
processKeystrokes () {
|
|
if (this.isEnabled()) {
|
|
this.zone.run(() => {
|
|
let matched = this.getCurrentFullyMatchedHotkey()
|
|
if (matched) {
|
|
console.log('Matched hotkey', matched)
|
|
this.matchedHotkey.emit(matched)
|
|
this.clearCurrentKeystrokes()
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
emitKeyEvent (nativeEvent) {
|
|
this.zone.run(() => {
|
|
this.key.emit(nativeEvent)
|
|
})
|
|
}
|
|
|
|
clearCurrentKeystrokes () {
|
|
this.currentKeystrokes = []
|
|
}
|
|
|
|
getCurrentKeystrokes (): string[] {
|
|
this.currentKeystrokes = this.currentKeystrokes.filter(x => performance.now() - x.time < KEY_TIMEOUT)
|
|
return stringifyKeySequence(this.currentKeystrokes.map(x => x.event))
|
|
}
|
|
|
|
registerGlobalHotkey () {
|
|
this.electron.globalShortcut.unregisterAll()
|
|
let value = this.config.store.hotkeys['toggle-window'] || []
|
|
if (typeof value === 'string') {
|
|
value = [value]
|
|
}
|
|
value.forEach(item => {
|
|
item = (typeof item === 'string') ? [item] : item
|
|
|
|
try {
|
|
this.electron.globalShortcut.register(item[0].replace(/-/g, '+'), () => {
|
|
this.globalHotkey.emit()
|
|
})
|
|
} catch (err) {
|
|
console.error('Could not register the global hotkey:', err)
|
|
}
|
|
})
|
|
}
|
|
|
|
getHotkeysConfig () {
|
|
return this.getHotkeysConfigRecursive(this.config.store.hotkeys)
|
|
}
|
|
|
|
getHotkeysConfigRecursive (branch) {
|
|
let keys = {}
|
|
for (let key in branch) {
|
|
let value = branch[key]
|
|
if (value instanceof Object && !(value instanceof Array)) {
|
|
let subkeys = this.getHotkeysConfigRecursive(value)
|
|
for (let subkey in subkeys) {
|
|
keys[key + '.' + subkey] = subkeys[subkey]
|
|
}
|
|
} else {
|
|
if (typeof value === 'string') {
|
|
value = [value]
|
|
}
|
|
if (value) {
|
|
value = value.map((item) => (typeof item === 'string') ? [item] : item)
|
|
keys[key] = value
|
|
}
|
|
}
|
|
}
|
|
return keys
|
|
}
|
|
|
|
getCurrentFullyMatchedHotkey (): string {
|
|
let currentStrokes = this.getCurrentKeystrokes()
|
|
let config = this.getHotkeysConfig()
|
|
for (let id in config) {
|
|
for (let sequence of config[id]) {
|
|
if (currentStrokes.length < sequence.length) {
|
|
continue
|
|
}
|
|
if (sequence.every(
|
|
(x, index) =>
|
|
x.toLowerCase() ===
|
|
currentStrokes[currentStrokes.length - sequence.length + index].toLowerCase()
|
|
)) {
|
|
return id
|
|
}
|
|
}
|
|
}
|
|
return null
|
|
}
|
|
|
|
getCurrentPartiallyMatchedHotkeys (): PartialHotkeyMatch[] {
|
|
let currentStrokes = this.getCurrentKeystrokes()
|
|
let config = this.getHotkeysConfig()
|
|
let result = []
|
|
for (let id in config) {
|
|
for (let sequence of config[id]) {
|
|
for (let matchLength = Math.min(currentStrokes.length, sequence.length); matchLength > 0; matchLength--) {
|
|
if (sequence.slice(0, matchLength).every(
|
|
(x, index) =>
|
|
x.toLowerCase() ===
|
|
currentStrokes[currentStrokes.length - matchLength + index].toLowerCase()
|
|
)) {
|
|
result.push({
|
|
matchedLength: matchLength,
|
|
id,
|
|
strokes: sequence
|
|
})
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
getHotkeyDescription (id: string): IHotkeyDescription {
|
|
return this.hotkeyDescriptions.filter((x) => x.id === id)[0]
|
|
}
|
|
|
|
enable () {
|
|
this.disabledLevel--
|
|
}
|
|
|
|
disable () {
|
|
this.disabledLevel++
|
|
}
|
|
|
|
isEnabled () {
|
|
return this.disabledLevel === 0
|
|
}
|
|
|
|
async getHotkeyDescriptions (): Promise<IHotkeyDescription[]> {
|
|
return (
|
|
await Promise.all(
|
|
this.config.enabledServices(this.hotkeyProviders)
|
|
.map(async x => x.provide ? x.provide() : x.hotkeys)
|
|
)
|
|
).reduce((a, b) => a.concat(b))
|
|
}
|
|
}
|
|
|
|
@Injectable()
|
|
export class AppHotkeyProvider extends HotkeyProvider {
|
|
hotkeys: IHotkeyDescription[] = [
|
|
{
|
|
id: 'new-window',
|
|
name: 'New window',
|
|
},
|
|
{
|
|
id: 'toggle-window',
|
|
name: 'Toggle terminal window',
|
|
},
|
|
{
|
|
id: 'toggle-fullscreen',
|
|
name: 'Toggle fullscreen mode',
|
|
},
|
|
{
|
|
id: 'rename-tab',
|
|
name: 'Rename Tab',
|
|
},
|
|
{
|
|
id: 'close-tab',
|
|
name: 'Close tab',
|
|
},
|
|
{
|
|
id: 'toggle-last-tab',
|
|
name: 'Toggle last tab',
|
|
},
|
|
{
|
|
id: 'next-tab',
|
|
name: 'Next tab',
|
|
},
|
|
{
|
|
id: 'previous-tab',
|
|
name: 'Previous tab',
|
|
},
|
|
{
|
|
id: 'tab-1',
|
|
name: 'Tab 1',
|
|
},
|
|
{
|
|
id: 'tab-2',
|
|
name: 'Tab 2',
|
|
},
|
|
{
|
|
id: 'tab-3',
|
|
name: 'Tab 3',
|
|
},
|
|
{
|
|
id: 'tab-4',
|
|
name: 'Tab 4',
|
|
},
|
|
{
|
|
id: 'tab-5',
|
|
name: 'Tab 5',
|
|
},
|
|
{
|
|
id: 'tab-6',
|
|
name: 'Tab 6',
|
|
},
|
|
{
|
|
id: 'tab-7',
|
|
name: 'Tab 7',
|
|
},
|
|
{
|
|
id: 'tab-8',
|
|
name: 'Tab 8',
|
|
},
|
|
{
|
|
id: 'tab-9',
|
|
name: 'Tab 9',
|
|
},
|
|
{
|
|
id: 'tab-10',
|
|
name: 'Tab 10',
|
|
},
|
|
]
|
|
|
|
async provide (): Promise<IHotkeyDescription[]> {
|
|
return this.hotkeys
|
|
}
|
|
}
|