feat: support more mouse buttons (#936)
* feat: support more mouse buttons (#834) * Remove unused style. * Retrigger CI * fix: more button click and texts * fix: the mouse button 4-8 naming * feat: allow extra mouse buttons if userConfig version >= 4.1.1 * fix: version comparison * fix: read correctly the right module informations
This commit is contained in:
@@ -15,7 +15,12 @@ export enum MouseActionParam {
|
|||||||
scrollLeft,
|
scrollLeft,
|
||||||
scrollRight,
|
scrollRight,
|
||||||
accelerate,
|
accelerate,
|
||||||
decelerate
|
decelerate,
|
||||||
|
button4,
|
||||||
|
button5,
|
||||||
|
button6,
|
||||||
|
button7,
|
||||||
|
button8
|
||||||
}
|
}
|
||||||
|
|
||||||
export class MouseAction extends KeyAction {
|
export class MouseAction extends KeyAction {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import { HardwareModuleInfo } from './hardware-module-info';
|
import { LeftModuleInfo } from './left-module-info';
|
||||||
|
import { RightModuleInfo } from './right-module-info';
|
||||||
|
|
||||||
export interface HardwareModules {
|
export interface HardwareModules {
|
||||||
leftModuleInfo?: HardwareModuleInfo;
|
leftModuleInfo?: LeftModuleInfo;
|
||||||
rightModuleInfo?: HardwareModuleInfo;
|
rightModuleInfo?: RightModuleInfo;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,8 +6,9 @@ export * from './app-start-info';
|
|||||||
export * from './configuration-reply';
|
export * from './configuration-reply';
|
||||||
export * from './version-information';
|
export * from './version-information';
|
||||||
export * from './device-connection-state';
|
export * from './device-connection-state';
|
||||||
|
export * from './left-module-info';
|
||||||
export * from './hardware-modules';
|
export * from './hardware-modules';
|
||||||
export * from './hardware-module-info';
|
export * from './right-module-info';
|
||||||
export * from './save-user-configuration-data';
|
export * from './save-user-configuration-data';
|
||||||
export * from './udev-rules-info';
|
export * from './udev-rules-info';
|
||||||
export * from './update-firmware-data';
|
export * from './update-firmware-data';
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
export interface HardwareModuleInfo {
|
export interface LeftModuleInfo {
|
||||||
firmwareVersion?: string;
|
firmwareVersion?: string;
|
||||||
moduleProtocolVersion?: string;
|
moduleProtocolVersion?: string;
|
||||||
}
|
}
|
||||||
7
packages/uhk-common/src/models/right-module-info.ts
Normal file
7
packages/uhk-common/src/models/right-module-info.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
export interface RightModuleInfo {
|
||||||
|
deviceProtocolVersion?: string;
|
||||||
|
hardwareConfigVersion?: string;
|
||||||
|
firmwareVersion?: string;
|
||||||
|
moduleProtocolVersion?: string;
|
||||||
|
userConfigVersion?: string;
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import { HardwareModuleInfo, LogService, UhkBuffer } from 'uhk-common';
|
import { LeftModuleInfo, LogService, RightModuleInfo, UhkBuffer } from 'uhk-common';
|
||||||
import { DataOption, KBoot, Properties, UsbPeripheral } from 'kboot';
|
import { DataOption, KBoot, Properties, UsbPeripheral } from 'kboot';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
@@ -248,7 +248,7 @@ export class UhkOperations {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async getLeftModuleVersionInfo(): Promise<HardwareModuleInfo> {
|
public async getLeftModuleVersionInfo(): Promise<LeftModuleInfo> {
|
||||||
try {
|
try {
|
||||||
this.logService.debug('[DeviceOperation] USB[T]: Read left module version information');
|
this.logService.debug('[DeviceOperation] USB[T]: Read left module version information');
|
||||||
|
|
||||||
@@ -277,7 +277,7 @@ export class UhkOperations {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
public async getRightModuleVersionInfo(): Promise<HardwareModuleInfo> {
|
public async getRightModuleVersionInfo(): Promise<RightModuleInfo> {
|
||||||
this.logService.debug('[DeviceOperation] USB[T]: Read right module version information');
|
this.logService.debug('[DeviceOperation] USB[T]: Read right module version information');
|
||||||
|
|
||||||
const command = Buffer.from([UsbCommand.GetProperty, DevicePropertyIds.ProtocolVersions]);
|
const command = Buffer.from([UsbCommand.GetProperty, DevicePropertyIds.ProtocolVersions]);
|
||||||
@@ -287,7 +287,11 @@ export class UhkOperations {
|
|||||||
uhkBuffer.readUInt8();
|
uhkBuffer.readUInt8();
|
||||||
|
|
||||||
return {
|
return {
|
||||||
firmwareVersion: `${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}`
|
firmwareVersion: `${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}`,
|
||||||
|
deviceProtocolVersion: `${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}`,
|
||||||
|
moduleProtocolVersion: `${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}`,
|
||||||
|
userConfigVersion: `${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}`,
|
||||||
|
hardwareConfigVersion: `${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}.${uhkBuffer.readUInt16()}`
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -38,6 +38,7 @@
|
|||||||
></layer-tab>
|
></layer-tab>
|
||||||
<mouse-tab #tab *ngSwitchCase="tabName.Mouse" class="popover-content"
|
<mouse-tab #tab *ngSwitchCase="tabName.Mouse" class="popover-content"
|
||||||
[defaultKeyAction]="defaultKeyAction"
|
[defaultKeyAction]="defaultKeyAction"
|
||||||
|
[extraMouseButtonsSupported]="extraMouseButtonsSupported$ | async"
|
||||||
(validAction)="setKeyActionValidState($event)"
|
(validAction)="setKeyActionValidState($event)"
|
||||||
></mouse-tab>
|
></mouse-tab>
|
||||||
<macro-tab #tab *ngSwitchCase="tabName.Macro" class="popover-content"
|
<macro-tab #tab *ngSwitchCase="tabName.Macro" class="popover-content"
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ import {
|
|||||||
|
|
||||||
import { Tab } from './tab';
|
import { Tab } from './tab';
|
||||||
|
|
||||||
import { AppState, getKeymaps, macroPlaybackSupported } from '../../store';
|
import { AppState, extraMouseButtonsSupported, getKeymaps, macroPlaybackSupported } from '../../store';
|
||||||
import { KeyActionRemap } from '../../models/key-action-remap';
|
import { KeyActionRemap } from '../../models/key-action-remap';
|
||||||
import { RemapInfo } from '../../models/remap-info';
|
import { RemapInfo } from '../../models/remap-info';
|
||||||
|
|
||||||
@@ -145,6 +145,7 @@ export class PopoverComponent implements OnChanges {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
macroPlaybackSupported$: Observable<boolean>;
|
macroPlaybackSupported$: Observable<boolean>;
|
||||||
|
extraMouseButtonsSupported$: Observable<boolean>;
|
||||||
|
|
||||||
private readonly currentKeymap$ = new BehaviorSubject<Keymap>(undefined);
|
private readonly currentKeymap$ = new BehaviorSubject<Keymap>(undefined);
|
||||||
|
|
||||||
@@ -158,6 +159,7 @@ export class PopoverComponent implements OnChanges {
|
|||||||
keymaps.filter((keymap: Keymap) => currentKeymap.abbreviation !== keymap.abbreviation))
|
keymaps.filter((keymap: Keymap) => currentKeymap.abbreviation !== keymap.abbreviation))
|
||||||
);
|
);
|
||||||
this.macroPlaybackSupported$ = store.select(macroPlaybackSupported);
|
this.macroPlaybackSupported$ = store.select(macroPlaybackSupported);
|
||||||
|
this.extraMouseButtonsSupported$ = store.select(extraMouseButtonsSupported);
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnChanges(change: SimpleChanges) {
|
ngOnChanges(change: SimpleChanges) {
|
||||||
|
|||||||
@@ -80,6 +80,27 @@
|
|||||||
[class.btn-primary]="mouseActionParam === MouseActionParam.rightClick"
|
[class.btn-primary]="mouseActionParam === MouseActionParam.rightClick"
|
||||||
(click)="setMouseActionParam(MouseActionParam.rightClick)">Right</button>
|
(click)="setMouseActionParam(MouseActionParam.rightClick)">Right</button>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="additional-keys" *ngIf="extraMouseButtonsSupported">
|
||||||
|
<div class="btn-group col-xs-12" role="group">
|
||||||
|
<button type="button" class="btn btn-default col-xs-4"
|
||||||
|
[class.btn-primary]="mouseActionParam === MouseActionParam.button4"
|
||||||
|
(click)="setMouseActionParam(MouseActionParam.button4)">Button 4</button>
|
||||||
|
<button type="button" class="btn btn-default col-xs-4"
|
||||||
|
[class.btn-primary]="mouseActionParam === MouseActionParam.button5"
|
||||||
|
(click)="setMouseActionParam(MouseActionParam.button5)">Button 5</button>
|
||||||
|
<button type="button" class="btn btn-default col-xs-4"
|
||||||
|
[class.btn-primary]="mouseActionParam === MouseActionParam.button6"
|
||||||
|
(click)="setMouseActionParam(MouseActionParam.button6)">Button 6</button>
|
||||||
|
</div>
|
||||||
|
<div class="btn-group col-xs-12" role="group">
|
||||||
|
<button type="button" class="btn btn-default col-xs-6"
|
||||||
|
[class.btn-primary]="mouseActionParam === MouseActionParam.button7"
|
||||||
|
(click)="setMouseActionParam(MouseActionParam.button7)">Button 7</button>
|
||||||
|
<button type="button" class="btn btn-default col-xs-6"
|
||||||
|
[class.btn-primary]="mouseActionParam === MouseActionParam.button8"
|
||||||
|
(click)="setMouseActionParam(MouseActionParam.button8)">Button 8</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div *ngSwitchCase="3" class="mouse__config mouse__config--speed text-center">
|
<div *ngSwitchCase="3" class="mouse__config mouse__config--speed text-center">
|
||||||
<div class="help-text--mouse-speed text-left">
|
<div class="help-text--mouse-speed text-left">
|
||||||
|
|||||||
@@ -54,6 +54,32 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.mouse__config--click {
|
||||||
|
.additional-keys {
|
||||||
|
display: inline-block;
|
||||||
|
margin-top: 1rem;
|
||||||
|
|
||||||
|
.btn-group:first-child {
|
||||||
|
.btn {
|
||||||
|
border-bottom-left-radius: 0;
|
||||||
|
border-bottom-right-radius: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-group:last-child {
|
||||||
|
.btn {
|
||||||
|
border-top: 0;
|
||||||
|
border-top-left-radius: 0;
|
||||||
|
border-top-right-radius: 0;
|
||||||
|
|
||||||
|
&:last-child {
|
||||||
|
right: 1px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.mouse__config--speed {
|
.mouse__config--speed {
|
||||||
.btn-default {
|
.btn-default {
|
||||||
font-size: 25px;
|
font-size: 25px;
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import { Tab } from '../tab';
|
|||||||
})
|
})
|
||||||
export class MouseTabComponent extends Tab implements OnChanges {
|
export class MouseTabComponent extends Tab implements OnChanges {
|
||||||
@Input() defaultKeyAction: KeyAction;
|
@Input() defaultKeyAction: KeyAction;
|
||||||
|
@Input() extraMouseButtonsSupported: boolean;
|
||||||
|
|
||||||
/* tslint:disable:variable-name: It is an enum type. So it can start with uppercase. */
|
/* tslint:disable:variable-name: It is an enum type. So it can start with uppercase. */
|
||||||
MouseActionParam = MouseActionParam;
|
MouseActionParam = MouseActionParam;
|
||||||
@@ -62,6 +63,11 @@ export class MouseTabComponent extends Tab implements OnChanges {
|
|||||||
case MouseActionParam.leftClick:
|
case MouseActionParam.leftClick:
|
||||||
case MouseActionParam.middleClick:
|
case MouseActionParam.middleClick:
|
||||||
case MouseActionParam.rightClick:
|
case MouseActionParam.rightClick:
|
||||||
|
case MouseActionParam.button4:
|
||||||
|
case MouseActionParam.button5:
|
||||||
|
case MouseActionParam.button6:
|
||||||
|
case MouseActionParam.button7:
|
||||||
|
case MouseActionParam.button8:
|
||||||
this.selectedPageIndex = 2;
|
this.selectedPageIndex = 2;
|
||||||
break;
|
break;
|
||||||
case MouseActionParam.decelerate:
|
case MouseActionParam.decelerate:
|
||||||
|
|||||||
@@ -28,6 +28,26 @@ export class SvgMouseKeyComponent implements OnChanges {
|
|||||||
this.type = 'click';
|
this.type = 'click';
|
||||||
this.param = 'Middle';
|
this.param = 'Middle';
|
||||||
break;
|
break;
|
||||||
|
case MouseActionParam.button4:
|
||||||
|
this.type = 'click';
|
||||||
|
this.param = 'Button 4';
|
||||||
|
break;
|
||||||
|
case MouseActionParam.button5:
|
||||||
|
this.type = 'click';
|
||||||
|
this.param = 'Button 5';
|
||||||
|
break;
|
||||||
|
case MouseActionParam.button6:
|
||||||
|
this.type = 'click';
|
||||||
|
this.param = 'Button 6';
|
||||||
|
break;
|
||||||
|
case MouseActionParam.button7:
|
||||||
|
this.type = 'click';
|
||||||
|
this.param = 'Button 7';
|
||||||
|
break;
|
||||||
|
case MouseActionParam.button8:
|
||||||
|
this.type = 'click';
|
||||||
|
this.param = 'Button 8';
|
||||||
|
break;
|
||||||
case MouseActionParam.scrollDown:
|
case MouseActionParam.scrollDown:
|
||||||
this.type = 'scroll';
|
this.type = 'scroll';
|
||||||
this.param = 'down';
|
this.param = 'down';
|
||||||
|
|||||||
@@ -157,3 +157,6 @@ export const layerDoubleTapSupported = createSelector(
|
|||||||
return isVersionGte(hardwareModules.rightModuleInfo.firmwareVersion, '8.4.3');
|
return isVersionGte(hardwareModules.rightModuleInfo.firmwareVersion, '8.4.3');
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
export const extraMouseButtonsSupported = createSelector(getHardwareModules, (hardwareModules: HardwareModules): boolean => {
|
||||||
|
return isVersionGte(hardwareModules.rightModuleInfo.userConfigVersion, '4.1.1');
|
||||||
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user