aboutsummaryrefslogtreecommitdiffstats
path: root/server
diff options
context:
space:
mode:
authorPhilippe Perrin <philippe.perrin@sonarsource.com>2022-08-10 19:06:16 +0200
committersonartech <sonartech@sonarsource.com>2022-08-17 20:03:09 +0000
commit94d62ab2f0a8aa66770ed6373481b4bf2574e6e9 (patch)
treedc3d619c06cde026a31accce52f5c5ebe399d121 /server
parent42145471c0db5b7ec917adf8fd68e7b519b724fe (diff)
downloadsonarqube-94d62ab2f0a8aa66770ed6373481b4bf2574e6e9.tar.gz
sonarqube-94d62ab2f0a8aa66770ed6373481b4bf2574e6e9.zip
SONAR-17169 Improve scrolling architecture in issues page
Diffstat (limited to 'server')
-rw-r--r--server/sonar-web/src/main/js/apps/issues/components/IssueSourceViewerScrollContext.tsx30
-rw-r--r--server/sonar-web/src/main/js/apps/issues/components/IssuesApp.tsx28
-rw-r--r--server/sonar-web/src/main/js/apps/issues/components/IssuesSourceViewer.tsx124
-rw-r--r--server/sonar-web/src/main/js/apps/issues/components/__tests__/__snapshots__/IssuesSourceViewer-test.tsx.snap44
-rw-r--r--server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/ComponentSourceSnippetGroupViewer.tsx53
-rw-r--r--server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/CrossComponentSourceViewer.tsx3
-rw-r--r--server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/__tests__/ComponentSourceSnippetGroupViewer-test.tsx9
-rw-r--r--server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/__tests__/CrossComponentSourceViewer-test.tsx1
-rw-r--r--server/sonar-web/src/main/js/apps/issues/utils.ts1
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/LineCode.tsx37
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/LineCode-test.tsx.snap6
-rw-r--r--server/sonar-web/src/main/js/components/issue/IssueMessageBox.tsx72
-rw-r--r--server/sonar-web/src/main/js/components/rules/RuleTabViewer.tsx2
13 files changed, 233 insertions, 177 deletions
diff --git a/server/sonar-web/src/main/js/apps/issues/components/IssueSourceViewerScrollContext.tsx b/server/sonar-web/src/main/js/apps/issues/components/IssueSourceViewerScrollContext.tsx
new file mode 100644
index 00000000000..9a4a1a23619
--- /dev/null
+++ b/server/sonar-web/src/main/js/apps/issues/components/IssueSourceViewerScrollContext.tsx
@@ -0,0 +1,30 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2022 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 React from 'react';
+
+export interface IssueSourceViewerScrollContextInterface {
+ registerPrimaryLocationRef: React.Ref<HTMLElement>;
+ registerSelectedSecondaryLocationRef: React.Ref<HTMLElement>;
+}
+
+export const IssueSourceViewerScrollContext = React.createContext<
+ IssueSourceViewerScrollContextInterface | undefined
+>(undefined);
diff --git a/server/sonar-web/src/main/js/apps/issues/components/IssuesApp.tsx b/server/sonar-web/src/main/js/apps/issues/components/IssuesApp.tsx
index 46fea9e8fb1..e1b7e1caaa5 100644
--- a/server/sonar-web/src/main/js/apps/issues/components/IssuesApp.tsx
+++ b/server/sonar-web/src/main/js/apps/issues/components/IssuesApp.tsx
@@ -89,7 +89,6 @@ import {
parseQuery,
Query,
saveMyIssues,
- scrollToIssue,
serializeQuery,
shouldOpenSonarSourceSecurityFacet,
shouldOpenStandardsChildFacet,
@@ -227,13 +226,6 @@ export class App extends React.PureComponent<Props, State> {
) {
this.fetchFirstIssues();
this.setState({ checkAll: false });
- } else if (
- !this.state.openIssue &&
- (prevState.selected !== this.state.selected || prevState.openIssue)
- ) {
- // if user simply selected another issue
- // or if user went from the source code back to the list of issues
- this.scrollToSelectedIssue();
} else if (openIssue && openIssue.key !== this.state.selected) {
this.setState({
locationsNavigator: false,
@@ -391,14 +383,11 @@ export class App extends React.PureComponent<Props, State> {
};
if (this.state.openIssue) {
if (path.query.open && path.query.open === this.state.openIssue.key) {
- this.setState(
- {
- locationsNavigator: false,
- selectedFlowIndex: undefined,
- selectedLocationIndex: undefined
- },
- this.scrollToSelectedIssue
- );
+ this.setState({
+ locationsNavigator: false,
+ selectedFlowIndex: undefined,
+ selectedLocationIndex: undefined
+ });
} else {
this.props.router.replace(path);
}
@@ -429,13 +418,6 @@ export class App extends React.PureComponent<Props, State> {
}
};
- scrollToSelectedIssue = (smooth = true) => {
- const { selected } = this.state;
- if (selected) {
- scrollToIssue(selected, smooth);
- }
- };
-
createdAfterIncludesTime = () => Boolean(this.props.location.query.createdAfter?.includes('T'));
fetchIssuesHelper = (query: RawQuery) => {
diff --git a/server/sonar-web/src/main/js/apps/issues/components/IssuesSourceViewer.tsx b/server/sonar-web/src/main/js/apps/issues/components/IssuesSourceViewer.tsx
index ee79ab27059..8e0999fd403 100644
--- a/server/sonar-web/src/main/js/apps/issues/components/IssuesSourceViewer.tsx
+++ b/server/sonar-web/src/main/js/apps/issues/components/IssuesSourceViewer.tsx
@@ -22,6 +22,7 @@ import { BranchLike } from '../../../types/branch-like';
import { Issue } from '../../../types/types';
import CrossComponentSourceViewer from '../crossComponentSourceViewer/CrossComponentSourceViewer';
import { getLocations, getSelectedLocation } from '../utils';
+import { IssueSourceViewerScrollContext } from './IssueSourceViewerScrollContext';
export interface IssuesSourceViewerProps {
branchLike: BranchLike | undefined;
@@ -34,39 +35,96 @@ export interface IssuesSourceViewerProps {
selectedLocationIndex: number | undefined;
}
-export default function IssuesSourceViewer(props: IssuesSourceViewerProps) {
- const {
- openIssue,
- selectedFlowIndex,
- selectedLocationIndex,
- locationsNavigator,
- branchLike,
- issues
- } = props;
+export default class IssuesSourceViewer extends React.PureComponent<IssuesSourceViewerProps> {
+ primaryLocationRef?: HTMLElement;
+ selectedSecondaryLocationRef?: HTMLElement;
- const locations = getLocations(openIssue, selectedFlowIndex).map((loc, index) => {
- loc.index = index;
- return loc;
- });
- const selectedLocation = getSelectedLocation(openIssue, selectedFlowIndex, selectedLocationIndex);
+ componentDidUpdate() {
+ if (this.props.selectedLocationIndex === -1) {
+ this.refreshScroll();
+ }
+ }
- const highlightedLocationMessage =
- locationsNavigator && selectedLocationIndex !== undefined
- ? selectedLocation && { index: selectedLocationIndex, text: selectedLocation.msg }
- : undefined;
- return (
- <div>
- <CrossComponentSourceViewer
- branchLike={branchLike}
- highlightedLocationMessage={highlightedLocationMessage}
- issue={openIssue}
- issues={issues}
- locations={locations}
- onIssueSelect={props.onIssueSelect}
- onLocationSelect={props.onLocationSelect}
- selectedFlowIndex={selectedFlowIndex}
- selectedLocationIndex={selectedLocationIndex}
- />
- </div>
- );
+ registerPrimaryLocationRef = (ref: HTMLElement) => {
+ this.primaryLocationRef = ref;
+
+ if (ref) {
+ this.refreshScroll();
+ }
+ };
+
+ registerSelectedSecondaryLocationRef = (ref: HTMLElement) => {
+ this.selectedSecondaryLocationRef = ref;
+
+ if (ref) {
+ this.refreshScroll();
+ }
+ };
+
+ refreshScroll() {
+ const { selectedLocationIndex } = this.props;
+
+ if (
+ selectedLocationIndex !== undefined &&
+ selectedLocationIndex !== -1 &&
+ this.selectedSecondaryLocationRef
+ ) {
+ this.selectedSecondaryLocationRef.scrollIntoView({
+ behavior: 'smooth',
+ block: 'center',
+ inline: 'nearest'
+ });
+ } else if (this.primaryLocationRef) {
+ this.primaryLocationRef.scrollIntoView({
+ block: 'center',
+ inline: 'nearest'
+ });
+ }
+ }
+
+ render() {
+ const {
+ openIssue,
+ selectedFlowIndex,
+ selectedLocationIndex,
+ locationsNavigator,
+ branchLike,
+ issues
+ } = this.props;
+
+ const locations = getLocations(openIssue, selectedFlowIndex).map((loc, index) => {
+ loc.index = index;
+ return loc;
+ });
+
+ const selectedLocation = getSelectedLocation(
+ openIssue,
+ selectedFlowIndex,
+ selectedLocationIndex
+ );
+
+ const highlightedLocationMessage =
+ locationsNavigator && selectedLocationIndex !== undefined
+ ? selectedLocation && { index: selectedLocationIndex, text: selectedLocation.msg }
+ : undefined;
+
+ return (
+ <IssueSourceViewerScrollContext.Provider
+ value={{
+ registerPrimaryLocationRef: this.registerPrimaryLocationRef,
+ registerSelectedSecondaryLocationRef: this.registerSelectedSecondaryLocationRef
+ }}>
+ <CrossComponentSourceViewer
+ branchLike={branchLike}
+ highlightedLocationMessage={highlightedLocationMessage}
+ issue={openIssue}
+ issues={issues}
+ locations={locations}
+ onIssueSelect={this.props.onIssueSelect}
+ onLocationSelect={this.props.onLocationSelect}
+ selectedFlowIndex={selectedFlowIndex}
+ />
+ </IssueSourceViewerScrollContext.Provider>
+ );
+ }
}
diff --git a/server/sonar-web/src/main/js/apps/issues/components/__tests__/__snapshots__/IssuesSourceViewer-test.tsx.snap b/server/sonar-web/src/main/js/apps/issues/components/__tests__/__snapshots__/IssuesSourceViewer-test.tsx.snap
index 17f3c6d7887..f835fb8a7b3 100644
--- a/server/sonar-web/src/main/js/apps/issues/components/__tests__/__snapshots__/IssuesSourceViewer-test.tsx.snap
+++ b/server/sonar-web/src/main/js/apps/issues/components/__tests__/__snapshots__/IssuesSourceViewer-test.tsx.snap
@@ -1,7 +1,14 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`should render CrossComponentSourceViewer correctly 1`] = `
-<div>
+<ContextProvider
+ value={
+ Object {
+ "registerPrimaryLocationRef": [Function],
+ "registerSelectedSecondaryLocationRef": [Function],
+ }
+ }
+>
<CrossComponentSourceViewer
branchLike={
Object {
@@ -211,11 +218,18 @@ exports[`should render CrossComponentSourceViewer correctly 1`] = `
onIssueSelect={[MockFunction]}
onLocationSelect={[MockFunction]}
/>
-</div>
+</ContextProvider>
`;
exports[`should render SourceViewer correctly: all secondary locations on same line 1`] = `
-<div>
+<ContextProvider
+ value={
+ Object {
+ "registerPrimaryLocationRef": [Function],
+ "registerSelectedSecondaryLocationRef": [Function],
+ }
+ }
+>
<CrossComponentSourceViewer
branchLike={
Object {
@@ -445,11 +459,18 @@ exports[`should render SourceViewer correctly: all secondary locations on same l
onIssueSelect={[MockFunction]}
onLocationSelect={[MockFunction]}
/>
-</div>
+</ContextProvider>
`;
exports[`should render SourceViewer correctly: default 1`] = `
-<div>
+<ContextProvider
+ value={
+ Object {
+ "registerPrimaryLocationRef": [Function],
+ "registerSelectedSecondaryLocationRef": [Function],
+ }
+ }
+>
<CrossComponentSourceViewer
branchLike={
Object {
@@ -525,11 +546,18 @@ exports[`should render SourceViewer correctly: default 1`] = `
onIssueSelect={[MockFunction]}
onLocationSelect={[MockFunction]}
/>
-</div>
+</ContextProvider>
`;
exports[`should render SourceViewer correctly: single secondary location 1`] = `
-<div>
+<ContextProvider
+ value={
+ Object {
+ "registerPrimaryLocationRef": [Function],
+ "registerSelectedSecondaryLocationRef": [Function],
+ }
+ }
+>
<CrossComponentSourceViewer
branchLike={
Object {
@@ -719,5 +747,5 @@ exports[`should render SourceViewer correctly: single secondary location 1`] = `
onIssueSelect={[MockFunction]}
onLocationSelect={[MockFunction]}
/>
-</div>
+</ContextProvider>
`;
diff --git a/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/ComponentSourceSnippetGroupViewer.tsx b/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/ComponentSourceSnippetGroupViewer.tsx
index 892849613c1..a8553151e0f 100644
--- a/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/ComponentSourceSnippetGroupViewer.tsx
+++ b/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/ComponentSourceSnippetGroupViewer.tsx
@@ -37,6 +37,7 @@ import {
SourceLine,
SourceViewerFile
} from '../../../types/types';
+import { IssueSourceViewerScrollContext } from '../components/IssueSourceViewerScrollContext';
import IssueSourceViewerHeader from './IssueSourceViewerHeader';
import SnippetViewer from './SnippetViewer';
import {
@@ -67,7 +68,6 @@ interface Props {
line: number
) => React.ReactNode;
snippetGroup: SnippetGroup;
- selectedLocationIndex: number | undefined;
}
interface State {
@@ -206,13 +206,7 @@ export default class ComponentSourceSnippetGroupViewer extends React.PureCompone
};
renderIssuesList = (line: SourceLine) => {
- const {
- isLastOccurenceOfPrimaryComponent,
- issue,
- issuesByLine,
- snippetGroup,
- selectedLocationIndex
- } = this.props;
+ const { isLastOccurenceOfPrimaryComponent, issue, issuesByLine, snippetGroup } = this.props;
const locations =
issue.component === snippetGroup.component.key && issue.textRange !== undefined
? locationsByLine([issue])
@@ -226,15 +220,21 @@ export default class ComponentSourceSnippetGroupViewer extends React.PureCompone
return (
issuesForLine.length > 0 && (
<div>
- {issuesForLine.map(issueToDisplay => (
- <IssueMessageBox
- selected={!!(issueToDisplay.key === issue.key && issueLocations.length > 0)}
- key={issueToDisplay.key}
- issue={issueToDisplay}
- onClick={this.props.onIssueSelect}
- selectedLocationIndex={selectedLocationIndex}
- />
- ))}
+ {issuesForLine.map(issueToDisplay => {
+ const isSelectedIssue = issueToDisplay.key === issue.key;
+ return (
+ <IssueSourceViewerScrollContext.Consumer key={issueToDisplay.key}>
+ {ctx => (
+ <IssueMessageBox
+ selected={!!(isSelectedIssue && issueLocations.length > 0)}
+ issue={issueToDisplay}
+ onClick={this.props.onIssueSelect}
+ ref={isSelectedIssue ? ctx?.registerPrimaryLocationRef : undefined}
+ />
+ )}
+ </IssueSourceViewerScrollContext.Consumer>
+ );
+ })}
</div>
)
);
@@ -246,8 +246,7 @@ export default class ComponentSourceSnippetGroupViewer extends React.PureCompone
isLastOccurenceOfPrimaryComponent,
issue,
lastSnippetGroup,
- snippetGroup,
- selectedLocationIndex
+ snippetGroup
} = this.props;
const { additionalLines, loading, snippets } = this.state;
const locations =
@@ -280,12 +279,16 @@ export default class ComponentSourceSnippetGroupViewer extends React.PureCompone
/>
{issue.component === snippetGroup.component.key && issue.textRange === undefined && (
- <IssueMessageBox
- selected={true}
- issue={issue}
- onClick={this.props.onIssueSelect}
- selectedLocationIndex={selectedLocationIndex}
- />
+ <IssueSourceViewerScrollContext.Consumer>
+ {ctx => (
+ <IssueMessageBox
+ selected={true}
+ issue={issue}
+ onClick={this.props.onIssueSelect}
+ ref={ctx?.registerPrimaryLocationRef}
+ />
+ )}
+ </IssueSourceViewerScrollContext.Consumer>
)}
{snippetLines.map((snippet, index) => (
<SnippetViewer
diff --git a/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/CrossComponentSourceViewer.tsx b/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/CrossComponentSourceViewer.tsx
index 77588f5fe94..6fcd02d3a24 100644
--- a/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/CrossComponentSourceViewer.tsx
+++ b/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/CrossComponentSourceViewer.tsx
@@ -62,7 +62,6 @@ interface Props {
onIssueSelect: (issueKey: string) => void;
onLocationSelect: (index: number) => void;
selectedFlowIndex: number | undefined;
- selectedLocationIndex: number | undefined;
}
interface State {
@@ -184,7 +183,6 @@ export default class CrossComponentSourceViewer extends React.PureComponent<Prop
render() {
const { loading, notAccessible } = this.state;
- const { selectedLocationIndex } = this.props;
if (loading) {
return (
@@ -238,7 +236,6 @@ export default class CrossComponentSourceViewer extends React.PureComponent<Prop
onLocationSelect={this.props.onLocationSelect}
renderDuplicationPopup={this.renderDuplicationPopup}
snippetGroup={snippetGroup}
- selectedLocationIndex={selectedLocationIndex}
/>
</SourceViewerContext.Provider>
);
diff --git a/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/__tests__/ComponentSourceSnippetGroupViewer-test.tsx b/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/__tests__/ComponentSourceSnippetGroupViewer-test.tsx
index 57e292a25d2..260173f7191 100644
--- a/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/__tests__/ComponentSourceSnippetGroupViewer-test.tsx
+++ b/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/__tests__/ComponentSourceSnippetGroupViewer-test.tsx
@@ -148,7 +148,13 @@ it('should render file-level issue correctly', () => {
}
});
- expect(wrapper.find(IssueMessageBox).exists()).toBe(true);
+ expect(
+ wrapper
+ .find('ContextConsumer')
+ .dive()
+ .find(IssueMessageBox)
+ .exists()
+ ).toBe(true);
});
it('should expand block', async () => {
@@ -294,7 +300,6 @@ function shallowRender(props: Partial<ComponentSourceSnippetGroupViewer['props']
<ComponentSourceSnippetGroupViewer
branchLike={mockMainBranch()}
highlightedLocationMessage={{ index: 0, text: '' }}
- selectedLocationIndex={0}
isLastOccurenceOfPrimaryComponent={true}
issue={mockIssue()}
issuesByLine={{}}
diff --git a/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/__tests__/CrossComponentSourceViewer-test.tsx b/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/__tests__/CrossComponentSourceViewer-test.tsx
index 22ba8be364b..645a2b9974e 100644
--- a/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/__tests__/CrossComponentSourceViewer-test.tsx
+++ b/server/sonar-web/src/main/js/apps/issues/crossComponentSourceViewer/__tests__/CrossComponentSourceViewer-test.tsx
@@ -116,7 +116,6 @@ function shallowRender(props: Partial<CrossComponentSourceViewer['props']> = {})
<CrossComponentSourceViewer
branchLike={undefined}
highlightedLocationMessage={undefined}
- selectedLocationIndex={undefined}
issue={mockIssue(true, {
key: '1',
component: 'project:main.js',
diff --git a/server/sonar-web/src/main/js/apps/issues/utils.ts b/server/sonar-web/src/main/js/apps/issues/utils.ts
index 2ace6b60664..d7266b3d8fe 100644
--- a/server/sonar-web/src/main/js/apps/issues/utils.ts
+++ b/server/sonar-web/src/main/js/apps/issues/utils.ts
@@ -236,6 +236,7 @@ export function allLocationsEmpty(
return getLocations(issue, selectedFlowIndex).every(location => !location.msg);
}
+// TODO: drop as it's useless now
export function scrollToIssue(issue: string, smooth = true) {
const element = document.querySelector(`[data-issue="${issue}"]`);
if (element) {
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/LineCode.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/LineCode.tsx
index 498b2df1e9a..713c0ea288b 100644
--- a/server/sonar-web/src/main/js/components/SourceViewer/components/LineCode.tsx
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/LineCode.tsx
@@ -19,6 +19,7 @@
*/
import classNames from 'classnames';
import * as React from 'react';
+import { IssueSourceViewerScrollContext } from '../../../apps/issues/components/IssueSourceViewerScrollContext';
import { LinearIssueLocation, SourceLine } from '../../../types/types';
import LocationIndex from '../../common/LocationIndex';
import Tooltip from '../../controls/Tooltip';
@@ -38,35 +39,8 @@ interface Props {
}
export default class LineCode extends React.PureComponent<React.PropsWithChildren<Props>> {
- activeMarkerNode?: HTMLElement | null;
symbols?: NodeListOf<HTMLElement>;
- componentDidMount() {
- if (this.activeMarkerNode) {
- this.activeMarkerNode.scrollIntoView({
- behavior: 'smooth',
- block: 'center',
- inline: 'center'
- });
- }
- }
-
- componentDidUpdate(prevProps: Props) {
- if (
- this.props.highlightedLocationMessage &&
- (!prevProps.highlightedLocationMessage ||
- prevProps.highlightedLocationMessage.index !==
- this.props.highlightedLocationMessage.index) &&
- this.activeMarkerNode
- ) {
- this.activeMarkerNode.scrollIntoView({
- behavior: 'smooth',
- block: 'center',
- inline: 'center'
- });
- }
- }
-
nodeNodeRef = (el: HTMLElement | null) => {
if (el) {
this.attachEvents(el);
@@ -105,7 +79,6 @@ export default class LineCode extends React.PureComponent<React.PropsWithChildre
renderMarker(index: number, message: string | undefined, selected: boolean, leading: boolean) {
const { onLocationSelect } = this.props;
const onClick = onLocationSelect ? () => onLocationSelect(index) : undefined;
- const ref = selected ? (node: HTMLElement | null) => (this.activeMarkerNode = node) : undefined;
return (
<Tooltip key={`marker-${index}`} overlay={message} placement="top">
@@ -114,7 +87,13 @@ export default class LineCode extends React.PureComponent<React.PropsWithChildre
onClick={onClick}
selected={selected}
aria-label={message ? `${index + 1}-${message}` : index + 1}>
- <span ref={ref}>{index + 1}</span>
+ <IssueSourceViewerScrollContext.Consumer>
+ {ctx => (
+ <span ref={selected ? ctx?.registerSelectedSecondaryLocationRef : undefined}>
+ {index + 1}
+ </span>
+ )}
+ </IssueSourceViewerScrollContext.Consumer>
</LocationIndex>
</Tooltip>
);
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/LineCode-test.tsx.snap b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/LineCode-test.tsx.snap
index fa518fddd0e..2dcef7709c8 100644
--- a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/LineCode-test.tsx.snap
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/LineCode-test.tsx.snap
@@ -117,9 +117,9 @@ exports[`render code: with secondary location 1`] = `
onClick={[Function]}
selected={false}
>
- <span>
- 2
- </span>
+ <ContextConsumer>
+ <Component />
+ </ContextConsumer>
</LocationIndex>
</Tooltip>
<span
diff --git a/server/sonar-web/src/main/js/components/issue/IssueMessageBox.tsx b/server/sonar-web/src/main/js/components/issue/IssueMessageBox.tsx
index cceb592fb35..e2b86d4edb5 100644
--- a/server/sonar-web/src/main/js/components/issue/IssueMessageBox.tsx
+++ b/server/sonar-web/src/main/js/components/issue/IssueMessageBox.tsx
@@ -28,56 +28,30 @@ export interface IssueMessageBoxProps {
selected: boolean;
issue: Issue;
onClick: (issueKey: string) => void;
- selectedLocationIndex?: number;
}
-export default class IssueMessageBox extends React.Component<IssueMessageBoxProps> {
- messageBoxRef: React.RefObject<HTMLDivElement> = React.createRef();
+export function IssueMessageBox(props: IssueMessageBoxProps, ref: React.ForwardedRef<any>) {
+ const { issue, selected } = props;
- componentDidMount() {
- if (this.props.selected && this.messageBoxRef.current) {
- this.messageBoxRef.current.scrollIntoView({
- block: 'center'
- });
- }
- }
-
- componentDidUpdate(prevProps: IssueMessageBoxProps) {
- if (
- this.messageBoxRef.current &&
- ((prevProps.selected !== this.props.selected && this.props.selected) ||
- (prevProps.selectedLocationIndex !== this.props.selectedLocationIndex &&
- this.props.selectedLocationIndex === -1))
- ) {
- this.messageBoxRef.current.scrollIntoView({
- block: 'center'
- });
- }
- }
-
- render() {
- const { issue, selected } = this.props;
- return (
- <div
- className={classNames(
- 'issue-message-box display-flex-row display-flex-center padded-right',
- {
- 'selected big-padded-top big-padded-bottom': selected,
- 'secondary-issue padded-top padded-bottom': !selected
- }
- )}
- key={issue.key}
- onClick={() => this.props.onClick(issue.key)}
- role="region"
- ref={this.messageBoxRef}
- aria-label={issue.message}>
- <IssueTypeIcon
- className="big-spacer-right spacer-left"
- fill={colors.baseFontColor}
- query={issue.type}
- />
- {issue.message}
- </div>
- );
- }
+ return (
+ <div
+ className={classNames('issue-message-box display-flex-row display-flex-center padded-right', {
+ 'selected big-padded-top big-padded-bottom': selected,
+ 'secondary-issue padded-top padded-bottom': !selected
+ })}
+ key={issue.key}
+ onClick={() => props.onClick(issue.key)}
+ role="region"
+ ref={ref}
+ aria-label={issue.message}>
+ <IssueTypeIcon
+ className="big-spacer-right spacer-left"
+ fill={colors.baseFontColor}
+ query={issue.type}
+ />
+ {issue.message}
+ </div>
+ );
}
+
+export default React.forwardRef(IssueMessageBox);
diff --git a/server/sonar-web/src/main/js/components/rules/RuleTabViewer.tsx b/server/sonar-web/src/main/js/components/rules/RuleTabViewer.tsx
index f4b9cdf4f1d..4adc421d9e3 100644
--- a/server/sonar-web/src/main/js/components/rules/RuleTabViewer.tsx
+++ b/server/sonar-web/src/main/js/components/rules/RuleTabViewer.tsx
@@ -119,7 +119,7 @@ export class RuleTabViewer extends React.PureComponent<RuleTabViewerProps, State
this.detachScrollEvent();
}
- computeState = (prevState: State, resetSelectedTab: boolean = false) => {
+ computeState = (prevState: State, resetSelectedTab = false) => {
const {
ruleDetails,
currentUser: { isLoggedIn, dismissedNotices }