aboutsummaryrefslogtreecommitdiffstats
path: root/server/sonar-web/src/main/js/components/hoc/withKeyboardNavigation.tsx
diff options
context:
space:
mode:
authorWouter Admiraal <wouter.admiraal@sonarsource.com>2019-01-17 08:50:30 +0100
committersonartech <sonartech@sonarsource.com>2019-02-11 09:11:24 +0100
commitb6aeddaea44525337d14ed3566fcd5f08d1e671f (patch)
treec282b7f8b88b6c945e507465aa321e52fb4c37b9 /server/sonar-web/src/main/js/components/hoc/withKeyboardNavigation.tsx
parent8b7cd93d7a751fd49e8df3faef1cf03e320f470a (diff)
downloadsonarqube-b6aeddaea44525337d14ed3566fcd5f08d1e671f.tar.gz
sonarqube-b6aeddaea44525337d14ed3566fcd5f08d1e671f.zip
SONAR-8697 Enable keyboard file navigation in Code page
Diffstat (limited to 'server/sonar-web/src/main/js/components/hoc/withKeyboardNavigation.tsx')
-rw-r--r--server/sonar-web/src/main/js/components/hoc/withKeyboardNavigation.tsx193
1 files changed, 193 insertions, 0 deletions
diff --git a/server/sonar-web/src/main/js/components/hoc/withKeyboardNavigation.tsx b/server/sonar-web/src/main/js/components/hoc/withKeyboardNavigation.tsx
new file mode 100644
index 00000000000..38f7cac5ebd
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/hoc/withKeyboardNavigation.tsx
@@ -0,0 +1,193 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2019 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+import * as React from 'react';
+import * as key from 'keymaster';
+import { getWrappedDisplayName } from './utils';
+import PageActions from '../ui/PageActions';
+
+export interface WithKeyboardNavigationProps {
+ components?: T.ComponentMeasure[];
+ cycle?: boolean;
+ isFile?: boolean;
+ onEndOfList?: () => void;
+ onGoToParent?: () => void;
+ onHighlight?: (item: T.ComponentMeasure) => void;
+ onSelect?: (item: T.ComponentMeasure) => void;
+ selected?: T.ComponentMeasure;
+}
+
+const KEY_SCOPE = 'key_nav';
+
+export default function withKeyboardNavigation<P>(
+ WrappedComponent: React.ComponentClass<P & Partial<WithKeyboardNavigationProps>>
+) {
+ return class Wrapper extends React.Component<P & WithKeyboardNavigationProps> {
+ static displayName = getWrappedDisplayName(WrappedComponent, 'withKeyboardNavigation');
+
+ componentDidMount() {
+ this.attachShortcuts();
+ }
+
+ componentWillUnmount() {
+ this.detachShortcuts();
+ }
+
+ attachShortcuts = () => {
+ key.setScope(KEY_SCOPE);
+ key('up', KEY_SCOPE, () => {
+ return this.skipIfFile(this.handleHighlightPrevious);
+ });
+ key('down', KEY_SCOPE, () => {
+ return this.skipIfFile(this.handleHighlightNext);
+ });
+ key('right,enter', KEY_SCOPE, () => {
+ return this.skipIfFile(this.handleSelectCurrent);
+ });
+ key('left', KEY_SCOPE, () => {
+ this.handleSelectParent();
+ return false; // always hijack left
+ });
+ key('k', KEY_SCOPE, () => {
+ return this.skipIfNotFile(this.handleSelectPrevious);
+ });
+ key('j', KEY_SCOPE, () => {
+ return this.skipIfNotFile(this.handleSelectNext);
+ });
+ };
+
+ detachShortcuts = () => {
+ key.deleteScope(KEY_SCOPE);
+ };
+
+ getCurrentIndex = () => {
+ const { selected, components = [] } = this.props;
+ return selected ? components.findIndex(component => component.key === selected.key) : -1;
+ };
+
+ skipIfFile = (handler: () => void) => {
+ if (this.props.isFile) {
+ return true;
+ } else {
+ handler();
+ return false;
+ }
+ };
+
+ skipIfNotFile = (handler: () => void) => {
+ if (this.props.isFile) {
+ handler();
+ return false;
+ } else {
+ return true;
+ }
+ };
+
+ handleHighlightNext = () => {
+ if (this.props.onHighlight === undefined) {
+ return;
+ }
+
+ const { components = [], cycle } = this.props;
+ const index = this.getCurrentIndex();
+ const first = cycle ? 0 : index;
+
+ this.props.onHighlight(
+ index < components.length - 1 ? components[index + 1] : components[first]
+ );
+
+ if (index + 1 === components.length - 1 && this.props.onEndOfList) {
+ this.props.onEndOfList();
+ }
+ };
+
+ handleHighlightPrevious = () => {
+ if (this.props.onHighlight === undefined) {
+ return;
+ }
+ const { components = [], cycle } = this.props;
+ const index = this.getCurrentIndex();
+ const last = cycle ? components.length - 1 : index;
+
+ this.props.onHighlight(index > 0 ? components[index - 1] : components[last]);
+ };
+
+ handleSelectCurrent = () => {
+ if (this.props.onSelect === undefined) {
+ return;
+ }
+
+ const { selected } = this.props;
+ if (selected !== undefined) {
+ this.props.onSelect(selected as T.ComponentMeasure);
+ }
+ };
+
+ handleSelectNext = () => {
+ if (this.props.onSelect === undefined) {
+ return;
+ }
+
+ const { components = [] } = this.props;
+ const index = this.getCurrentIndex();
+
+ if (index !== -1 && index < components.length - 1) {
+ this.props.onSelect(components[index + 1]);
+ }
+ };
+
+ handleSelectParent = () => {
+ if (this.props.onGoToParent !== undefined) {
+ this.props.onGoToParent();
+ }
+ };
+
+ handleSelectPrevious = () => {
+ if (this.props.onSelect === undefined) {
+ return;
+ }
+
+ const { components = [] } = this.props;
+ const index = this.getCurrentIndex();
+
+ if (components.length && index > 0) {
+ this.props.onSelect(components[index - 1]);
+ }
+ };
+
+ render() {
+ const { components = [], isFile } = this.props;
+ const index = this.getCurrentIndex();
+
+ return (
+ <>
+ <PageActions
+ current={index > -1 ? index + 1 : undefined}
+ isFile={isFile}
+ showPaging={isFile && index > -1}
+ showShortcuts={true}
+ totalLoadedComponents={components.length}
+ />
+
+ <WrappedComponent {...this.props} />
+ </>
+ );
+ }
+ };
+}