forked from MapComplete/MapComplete
		
	
		
			
				
	
	
		
			124 lines
		
	
	
	
		
			4.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			124 lines
		
	
	
	
		
			4.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { Utils } from "../../Utils"
 | |
| import Combine from "./Combine"
 | |
| import BaseUIElement from "../BaseUIElement"
 | |
| import Title from "./Title"
 | |
| import Table from "./Table"
 | |
| import { UIEventSource } from "../../Logic/UIEventSource"
 | |
| import { VariableUiElement } from "./VariableUIElement"
 | |
| import { Translation } from "../i18n/Translation"
 | |
| import { FixedUiElement } from "./FixedUiElement"
 | |
| 
 | |
| export default class Hotkeys {
 | |
|     private static readonly _docs: UIEventSource<
 | |
|         {
 | |
|             key: { ctrl?: string; shift?: string; alt?: string; nomod?: string; onUp?: boolean }
 | |
|             documentation: string | Translation
 | |
|         }[]
 | |
|     > = new UIEventSource<
 | |
|         {
 | |
|             key: { ctrl?: string; shift?: string; alt?: string; nomod?: string; onUp?: boolean }
 | |
|             documentation: string | Translation
 | |
|         }[]
 | |
|     >([])
 | |
| 
 | |
|     private static textElementSelected(): boolean {
 | |
|         return ["input", "textarea"].includes(document?.activeElement?.tagName?.toLowerCase())
 | |
|     }
 | |
|     public static RegisterHotkey(
 | |
|         key: (
 | |
|             | {
 | |
|                   ctrl: string
 | |
|               }
 | |
|             | {
 | |
|                   shift: string
 | |
|               }
 | |
|             | {
 | |
|                   alt: string
 | |
|               }
 | |
|             | {
 | |
|                   nomod: string
 | |
|               }
 | |
|         ) & {
 | |
|             onUp?: boolean
 | |
|         },
 | |
|         documentation: string | Translation,
 | |
|         action: () => void
 | |
|     ) {
 | |
|         const type = key["onUp"] ? "keyup" : "keypress"
 | |
|         let keycode: string = key["ctrl"] ?? key["shift"] ?? key["alt"] ?? key["nomod"]
 | |
|         if (keycode.length == 1) {
 | |
|             keycode = keycode.toLowerCase()
 | |
|             if (key["shift"] !== undefined) {
 | |
|                 keycode = keycode.toUpperCase()
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         this._docs.data.push({ key, documentation })
 | |
|         this._docs.ping()
 | |
|         if (Utils.runningFromConsole) {
 | |
|             return
 | |
|         }
 | |
|         if (key["ctrl"] !== undefined) {
 | |
|             document.addEventListener("keydown", function (event) {
 | |
|                 if (event.ctrlKey && event.key === keycode) {
 | |
|                     action()
 | |
|                     event.preventDefault()
 | |
|                 }
 | |
|             })
 | |
|         } else if (key["shift"] !== undefined) {
 | |
|             document.addEventListener(type, function (event) {
 | |
|                 if (Hotkeys.textElementSelected()) {
 | |
|                     // A text element is selected, we don't do anything special
 | |
|                     return
 | |
|                 }
 | |
|                 if (event.shiftKey && event.key === keycode) {
 | |
|                     action()
 | |
|                     event.preventDefault()
 | |
|                 }
 | |
|             })
 | |
|         } else if (key["alt"] !== undefined) {
 | |
|             document.addEventListener(type, function (event) {
 | |
|                 if (event.altKey && event.key === keycode) {
 | |
|                     action()
 | |
|                     event.preventDefault()
 | |
|                 }
 | |
|             })
 | |
|         } else if (key["nomod"] !== undefined) {
 | |
|             document.addEventListener(type, function (event) {
 | |
|                 if (Hotkeys.textElementSelected()) {
 | |
|                     // A text element is selected, we don't do anything special
 | |
|                     return
 | |
|                 }
 | |
|                 if (event.key === keycode) {
 | |
|                     action()
 | |
|                     event.preventDefault()
 | |
|                 }
 | |
|             })
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     static generateDocumentation(): BaseUIElement {
 | |
|         const byKey: [string, string | Translation][] = Hotkeys._docs.data
 | |
|             .map(({ key, documentation }) => {
 | |
|                 const modifiers = Object.keys(key).filter((k) => k !== "nomod" && k !== "onUp")
 | |
|                 const keycode: string = key["ctrl"] ?? key["shift"] ?? key["alt"] ?? key["nomod"]
 | |
|                 modifiers.push(keycode)
 | |
|                 return <[string, string | Translation]>[modifiers.join("+"), documentation]
 | |
|             })
 | |
|             .sort()
 | |
|         return new Combine([
 | |
|             new Title("Hotkeys", 1),
 | |
|             "MapComplete supports the following keys:",
 | |
|             new Table(
 | |
|                 ["Key combination", "Action"],
 | |
|                 byKey.map(([key, doc]) => {
 | |
|                     return [new FixedUiElement(key).SetClass("code"), doc]
 | |
|                 })
 | |
|             ),
 | |
|         ])
 | |
|     }
 | |
| 
 | |
|     static generateDocumentationDynamic(): BaseUIElement {
 | |
|         return new VariableUiElement(Hotkeys._docs.map((_) => Hotkeys.generateDocumentation()))
 | |
|     }
 | |
| }
 |