Skip to content

Commit 34e0179

Browse files
authored
Merge pull request #5794 from anthonykim1/anthonykim1/optionFBKitty
Fix macOS Option+key sending wrong codepoint in Kitty keyboard protocol
2 parents 4316263 + bd722e7 commit 34e0179

File tree

3 files changed

+89
-6
lines changed

3 files changed

+89
-6
lines changed

src/browser/services/KeyboardService.ts

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,7 @@ export class KeyboardService implements IKeyboardService {
4040
}
4141
const kittyFlags = this._coreService.kittyKeyboard.flags;
4242
return this.useKitty
43-
? this._getKittyKeyboard().evaluate(event, kittyFlags, event.repeat ? KittyKeyboardEventType.REPEAT : KittyKeyboardEventType.PRESS)
43+
? this._getKittyKeyboard().evaluate(event, kittyFlags, event.repeat ? KittyKeyboardEventType.REPEAT : KittyKeyboardEventType.PRESS, isMac && this._optionsService.rawOptions.macOptionIsMeta)
4444
: evaluateKeyboardEvent(event, this._coreService.decPrivateModes.applicationCursorKeys, isMac, this._optionsService.rawOptions.macOptionIsMeta);
4545
}
4646

@@ -51,7 +51,7 @@ export class KeyboardService implements IKeyboardService {
5151
}
5252
const kittyFlags = this._coreService.kittyKeyboard.flags;
5353
if (this.useKitty && (kittyFlags & KittyKeyboardFlags.REPORT_EVENT_TYPES)) {
54-
return this._getKittyKeyboard().evaluate(event, kittyFlags, KittyKeyboardEventType.RELEASE);
54+
return this._getKittyKeyboard().evaluate(event, kittyFlags, KittyKeyboardEventType.RELEASE, isMac && this._optionsService.rawOptions.macOptionIsMeta);
5555
}
5656
return undefined;
5757
}

src/common/input/KittyKeyboard.test.ts

Lines changed: 81 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -729,5 +729,86 @@ describe('KittyKeyboard', () => {
729729
assert.strictEqual(result.key, '\x1b[57440u');
730730
});
731731
});
732+
733+
describe('macOS Option as Alt (macOptionIsMeta)', () => {
734+
const flags = KittyKeyboardFlags.DISAMBIGUATE_ESCAPE_CODES;
735+
const press = KittyKeyboardEventType.PRESS;
736+
737+
it('Opt+f (key=ƒ) → CSI 102;3 u', () => {
738+
const result = kitty.evaluate(createEvent({ key: 'ƒ', code: 'KeyF', altKey: true }), flags, press, true);
739+
assert.strictEqual(result.key, '\x1b[102;3u');
740+
});
741+
742+
it('Opt+b (key=∫) → CSI 98;3 u', () => {
743+
const result = kitty.evaluate(createEvent({ key: '∫', code: 'KeyB', altKey: true }), flags, press, true);
744+
assert.strictEqual(result.key, '\x1b[98;3u');
745+
});
746+
747+
it('Opt+d (key=∂) → CSI 100;3 u', () => {
748+
const result = kitty.evaluate(createEvent({ key: '∂', code: 'KeyD', altKey: true }), flags, press, true);
749+
assert.strictEqual(result.key, '\x1b[100;3u');
750+
});
751+
752+
it('Opt+n dead key (key=Dead, code=KeyN) → CSI 110;3 u', () => {
753+
const result = kitty.evaluate(createEvent({ key: 'Dead', code: 'KeyN', altKey: true }), flags, press, true);
754+
assert.strictEqual(result.key, '\x1b[110;3u');
755+
});
756+
757+
it('Opt+e dead key (key=Dead, code=KeyE) → CSI 101;3 u', () => {
758+
const result = kitty.evaluate(createEvent({ key: 'Dead', code: 'KeyE', altKey: true }), flags, press, true);
759+
assert.strictEqual(result.key, '\x1b[101;3u');
760+
});
761+
762+
it('Opt+u dead key (key=Dead, code=KeyU) → CSI 117;3 u', () => {
763+
const result = kitty.evaluate(createEvent({ key: 'Dead', code: 'KeyU', altKey: true }), flags, press, true);
764+
assert.strictEqual(result.key, '\x1b[117;3u');
765+
});
766+
767+
it('Opt+5 (key=∞) → CSI 53;3 u', () => {
768+
const result = kitty.evaluate(createEvent({ key: '∞', code: 'Digit5', altKey: true }), flags, press, true);
769+
assert.strictEqual(result.key, '\x1b[53;3u');
770+
});
771+
772+
it('Opt+Shift+f (key=Ï) → CSI 102;4 u', () => {
773+
const result = kitty.evaluate(createEvent({ key: 'Ï', code: 'KeyF', altKey: true, shiftKey: true }), flags, press, true);
774+
assert.strictEqual(result.key, '\x1b[102;4u');
775+
});
776+
777+
it('Ctrl+Opt+f (key=ƒ) → CSI 102;7 u', () => {
778+
const result = kitty.evaluate(createEvent({ key: 'ƒ', code: 'KeyF', altKey: true, ctrlKey: true }), flags, press, true);
779+
assert.strictEqual(result.key, '\x1b[102;7u');
780+
});
781+
782+
it('does not unwind when macOptionAsAlt is false (Linux Alt is a chord)', () => {
783+
const result = kitty.evaluate(createEvent({ key: 'a', code: 'KeyA', altKey: true }), flags, press, false);
784+
assert.strictEqual(result.key, '\x1b[97;3u');
785+
});
786+
787+
it('does not unwind on Linux AZERTY (key=a, code=KeyQ) — uses ev.key not ev.code', () => {
788+
const result = kitty.evaluate(createEvent({ key: 'a', code: 'KeyQ', altKey: true }), flags, press, false);
789+
assert.strictEqual(result.key, '\x1b[97;3u');
790+
});
791+
792+
it('does not unwind when macOptionAsAlt is false even with composed key', () => {
793+
const result = kitty.evaluate(createEvent({ key: 'ƒ', code: 'KeyF', altKey: true }), flags, press, false);
794+
assert.strictEqual(result.key, '\x1b[402;3u');
795+
});
796+
797+
it('does not unwind when altKey is false', () => {
798+
const result = kitty.evaluate(createEvent({ key: 'ƒ', code: 'KeyF' }), flags, press, true);
799+
assert.strictEqual(result.key, 'ƒ');
800+
});
801+
802+
it('falls through when ev.code is not Key*/Digit* (Opt+;)', () => {
803+
const result = kitty.evaluate(createEvent({ key: '…', code: 'Semicolon', altKey: true }), flags, press, true);
804+
assert.strictEqual(result.key, '\x1b[8230;3u');
805+
});
806+
807+
it('Opt+f release with REPORT_EVENT_TYPES → CSI 102;3:3 u', () => {
808+
const releaseFlags = KittyKeyboardFlags.DISAMBIGUATE_ESCAPE_CODES | KittyKeyboardFlags.REPORT_EVENT_TYPES;
809+
const result = kitty.evaluate(createEvent({ key: 'ƒ', code: 'KeyF', altKey: true }), releaseFlags, KittyKeyboardEventType.RELEASE, true);
810+
assert.strictEqual(result.key, '\x1b[102;3:3u');
811+
});
812+
});
732813
});
733814
});

src/common/input/KittyKeyboard.ts

Lines changed: 6 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -218,7 +218,7 @@ export class KittyKeyboard {
218218
* Returns the lowercase codepoint for letters.
219219
* For shifted keys, uses the code property to get the base key.
220220
*/
221-
private _getKeyCode(ev: IKeyboardEvent): number | undefined {
221+
private _getKeyCode(ev: IKeyboardEvent, macOptionAsAlt: boolean): number | undefined {
222222
const numpadCode = this._getNumpadKeyCode(ev);
223223
if (numpadCode !== undefined) {
224224
return numpadCode;
@@ -234,7 +234,7 @@ export class KittyKeyboard {
234234
return funcCode;
235235
}
236236

237-
if (ev.shiftKey && ev.code) {
237+
if ((ev.shiftKey || (macOptionAsAlt && ev.altKey)) && ev.code) {
238238
if (ev.code.startsWith('Digit') && ev.code.length === 6) {
239239
const digit = ev.code.charAt(5);
240240
if (digit >= '0' && digit <= '9') {
@@ -410,12 +410,14 @@ export class KittyKeyboard {
410410
* @param ev The keyboard event.
411411
* @param flags The active Kitty keyboard enhancement flags.
412412
* @param eventType The event type (press, repeat, release).
413+
* @param macOptionAsAlt When true, macOS Option-composed ev.key values are unwound via ev.code.
413414
* @returns The keyboard result with the encoded key sequence.
414415
*/
415416
public evaluate(
416417
ev: IKeyboardEvent,
417418
flags: number,
418-
eventType: KittyKeyboardEventType = KittyKeyboardEventType.PRESS
419+
eventType: KittyKeyboardEventType = KittyKeyboardEventType.PRESS,
420+
macOptionAsAlt: boolean = false
419421
): IKeyboardResult {
420422
const result: IKeyboardResult = {
421423
type: KeyboardResultType.SEND_KEY,
@@ -464,7 +466,7 @@ export class KittyKeyboard {
464466
return result;
465467
}
466468

467-
const keyCode = this._getKeyCode(ev);
469+
const keyCode = this._getKeyCode(ev, macOptionAsAlt);
468470
if (keyCode === undefined) {
469471
return result;
470472
}

0 commit comments

Comments
 (0)