aboutsummaryrefslogtreecommitdiffstats
path: root/server
diff options
context:
space:
mode:
authorGrégoire Aubert <gregoire.aubert@sonarsource.com>2017-09-29 15:03:12 +0200
committerGrégoire Aubert <gregoire.aubert@sonarsource.com>2017-09-29 17:09:48 +0200
commit059cf8688eafd9f151a1fad84997ce77e5af511e (patch)
treeb0d0941506386d23955ee9b893c8cc73e0c07b2d /server
parentee41573b8b309a123bc23ac623663107cc410af3 (diff)
downloadsonarqube-059cf8688eafd9f151a1fad84997ce77e5af511e.tar.gz
sonarqube-059cf8688eafd9f151a1fad84997ce77e5af511e.zip
SONAR-9792 Fix sidebar position with a notification
Diffstat (limited to 'server')
-rw-r--r--server/sonar-web/src/main/js/apps/component-measures/components/App.js25
-rw-r--r--server/sonar-web/src/main/js/apps/component-measures/components/__tests__/__snapshots__/App-test.js.snap27
-rw-r--r--server/sonar-web/src/main/js/apps/issues/components/App.js17
-rw-r--r--server/sonar-web/src/main/js/apps/quality-gates/components/QualityGatesApp.js21
-rw-r--r--server/sonar-web/src/main/js/apps/web-api/components/WebApiApp.tsx55
-rw-r--r--server/sonar-web/src/main/js/components/common/ScreenPositionHelper.tsx67
6 files changed, 132 insertions, 80 deletions
diff --git a/server/sonar-web/src/main/js/apps/component-measures/components/App.js b/server/sonar-web/src/main/js/apps/component-measures/components/App.js
index a9fc88958ad..a33340530ea 100644
--- a/server/sonar-web/src/main/js/apps/component-measures/components/App.js
+++ b/server/sonar-web/src/main/js/apps/component-measures/components/App.js
@@ -24,6 +24,7 @@ import key from 'keymaster';
import MeasureContentContainer from './MeasureContentContainer';
import MeasureOverviewContainer from './MeasureOverviewContainer';
import Sidebar from '../sidebar/Sidebar';
+import ScreenPositionHelper from '../../../components/common/ScreenPositionHelper';
import { hasBubbleChart, parseQuery, serializeQuery } from '../utils';
import { getBranchName } from '../../../helpers/branches';
import { translate } from '../../../helpers/l10n';
@@ -148,19 +149,21 @@ export default class App extends React.PureComponent {
<div className="layout-page" id="component-measures">
<Helmet title={translate('layout.measures')} />
- <div className="layout-page-side-outer">
- <div className="layout-page-side" style={{ top: 95 }}>
- <div className="layout-page-side-inner">
- <div className="layout-page-filters">
- <Sidebar
- measures={this.state.measures}
- selectedMetric={query.metric}
- updateQuery={this.updateQuery}
- />
+ <ScreenPositionHelper className="layout-page-side-outer">
+ {({ top }) => (
+ <div className="layout-page-side" style={{ top }}>
+ <div className="layout-page-side-inner">
+ <div className="layout-page-filters">
+ <Sidebar
+ measures={this.state.measures}
+ selectedMetric={query.metric}
+ updateQuery={this.updateQuery}
+ />
+ </div>
</div>
</div>
- </div>
- </div>
+ )}
+ </ScreenPositionHelper>
{metric != null && (
<MeasureContentContainer
diff --git a/server/sonar-web/src/main/js/apps/component-measures/components/__tests__/__snapshots__/App-test.js.snap b/server/sonar-web/src/main/js/apps/component-measures/components/__tests__/__snapshots__/App-test.js.snap
index 85d80ad9d74..24354987bd6 100644
--- a/server/sonar-web/src/main/js/apps/component-measures/components/__tests__/__snapshots__/App-test.js.snap
+++ b/server/sonar-web/src/main/js/apps/component-measures/components/__tests__/__snapshots__/App-test.js.snap
@@ -9,32 +9,9 @@ exports[`should render correctly 1`] = `
encodeSpecialCharacters={true}
title="layout.measures"
/>
- <div
+ <ScreenPositionHelper
className="layout-page-side-outer"
- >
- <div
- className="layout-page-side"
- style={
- Object {
- "top": 95,
- }
- }
- >
- <div
- className="layout-page-side-inner"
- >
- <div
- className="layout-page-filters"
- >
- <Sidebar
- measures={Array []}
- selectedMetric="coverage"
- updateQuery={[Function]}
- />
- </div>
- </div>
- </div>
- </div>
+ />
<MeasureContentContainer
className="layout-page-main"
fetchMeasures={[Function]}
diff --git a/server/sonar-web/src/main/js/apps/issues/components/App.js b/server/sonar-web/src/main/js/apps/issues/components/App.js
index 888ce25b468..f5f83df90e7 100644
--- a/server/sonar-web/src/main/js/apps/issues/components/App.js
+++ b/server/sonar-web/src/main/js/apps/issues/components/App.js
@@ -55,6 +55,7 @@ import {
} from '../utils'; */
import ListFooter from '../../../components/controls/ListFooter';
import EmptySearch from '../../../components/common/EmptySearch';
+import ScreenPositionHelper from '../../../components/common/ScreenPositionHelper';
import { getBranchName } from '../../../helpers/branches';
import { translate, translateWithParameters } from '../../../helpers/l10n';
import { scrollToElement } from '../../../helpers/scrolling';
@@ -792,16 +793,16 @@ export default class App extends React.PureComponent {
}
renderSide(openIssue /*: ?Issue */) {
- const top = this.props.component || this.props.organization ? 95 : 30;
-
return (
- <div className="layout-page-side-outer">
- <div className="layout-page-side" style={{ top }}>
- <div className="layout-page-side-inner">
- {openIssue == null ? this.renderFacets() : this.renderConciseIssuesList()}
+ <ScreenPositionHelper className="layout-page-side-outer">
+ {({ top }) => (
+ <div className="layout-page-side" style={{ top }}>
+ <div className="layout-page-side-inner">
+ {openIssue == null ? this.renderFacets() : this.renderConciseIssuesList()}
+ </div>
</div>
- </div>
- </div>
+ )}
+ </ScreenPositionHelper>
);
}
diff --git a/server/sonar-web/src/main/js/apps/quality-gates/components/QualityGatesApp.js b/server/sonar-web/src/main/js/apps/quality-gates/components/QualityGatesApp.js
index 1499f139787..b756fee0c39 100644
--- a/server/sonar-web/src/main/js/apps/quality-gates/components/QualityGatesApp.js
+++ b/server/sonar-web/src/main/js/apps/quality-gates/components/QualityGatesApp.js
@@ -22,6 +22,7 @@ import PropTypes from 'prop-types';
import Helmet from 'react-helmet';
import ListHeader from './ListHeader';
import List from './List';
+import ScreenPositionHelper from '../../../components/common/ScreenPositionHelper';
import {
fetchQualityGatesAppDetails,
fetchQualityGates as fetchQualityGatesAPI
@@ -67,22 +68,22 @@ export default class QualityGatesApp extends Component {
render() {
const { children, qualityGates, edit, organization } = this.props;
const defaultTitle = translate('quality_gates.page');
- const top = organization ? 95 : 30;
return (
<div id="quality-gates-page" className="layout-page">
<Helmet defaultTitle={defaultTitle} titleTemplate={'%s - ' + defaultTitle} />
- <div className="layout-page-side-outer">
- <div className="layout-page-side" style={{ top }}>
- <div className="layout-page-side-inner">
- <div className="layout-page-filters">
- <ListHeader canEdit={edit} onAdd={this.handleAdd.bind(this)} />
- {qualityGates && <List organization={organization} qualityGates={qualityGates} />}
+ <ScreenPositionHelper className="layout-page-side-outer">
+ {({ top }) => (
+ <div className="layout-page-side" style={{ top }}>
+ <div className="layout-page-side-inner">
+ <div className="layout-page-filters">
+ <ListHeader canEdit={edit} onAdd={this.handleAdd.bind(this)} />
+ {qualityGates && <List organization={organization} qualityGates={qualityGates} />}
+ </div>
</div>
</div>
- </div>
- </div>
-
+ )}
+ </ScreenPositionHelper>
{qualityGates != null &&
React.Children.map(children, child => React.cloneElement(child, { organization }))}
</div>
diff --git a/server/sonar-web/src/main/js/apps/web-api/components/WebApiApp.tsx b/server/sonar-web/src/main/js/apps/web-api/components/WebApiApp.tsx
index 3a69c4bc4c6..ebc579d5ca5 100644
--- a/server/sonar-web/src/main/js/apps/web-api/components/WebApiApp.tsx
+++ b/server/sonar-web/src/main/js/apps/web-api/components/WebApiApp.tsx
@@ -25,6 +25,7 @@ import { Domain as DomainType, fetchWebApi } from '../../../api/web-api';
import Menu from './Menu';
import Search from './Search';
import Domain from './Domain';
+import ScreenPositionHelper from '../../../components/common/ScreenPositionHelper';
import { getActionKey, isDomainPathActive } from '../utils';
import { scrollToElement } from '../../../helpers/scrolling';
import { translate } from '../../../helpers/l10n';
@@ -141,35 +142,37 @@ export default class WebApiApp extends React.PureComponent<Props, State> {
return (
<div className="layout-page">
<Helmet title={translate('api_documentation.page')} />
- <div className="layout-page-side-outer">
- <div className="layout-page-side" style={{ top: 30 }}>
- <div className="layout-page-side-inner">
- <div className="layout-page-filters">
- <div className="web-api-page-header">
- <Link to="/web_api/">
- <h1>{translate('api_documentation.page')}</h1>
- </Link>
+ <ScreenPositionHelper className="layout-page-side-outer">
+ {({ top }) => (
+ <div className="layout-page-side" style={{ top }}>
+ <div className="layout-page-side-inner">
+ <div className="layout-page-filters">
+ <div className="web-api-page-header">
+ <Link to="/web_api/">
+ <h1>{translate('api_documentation.page')}</h1>
+ </Link>
+ </div>
+
+ <Search
+ showDeprecated={showDeprecated}
+ showInternal={showInternal}
+ onSearch={this.handleSearch}
+ onToggleInternal={this.handleToggleInternal}
+ onToggleDeprecated={this.handleToggleDeprecated}
+ />
+
+ <Menu
+ domains={this.state.domains}
+ showDeprecated={showDeprecated}
+ showInternal={showInternal}
+ searchQuery={searchQuery}
+ splat={splat}
+ />
</div>
-
- <Search
- showDeprecated={showDeprecated}
- showInternal={showInternal}
- onSearch={this.handleSearch}
- onToggleInternal={this.handleToggleInternal}
- onToggleDeprecated={this.handleToggleDeprecated}
- />
-
- <Menu
- domains={this.state.domains}
- showDeprecated={showDeprecated}
- showInternal={showInternal}
- searchQuery={searchQuery}
- splat={splat}
- />
</div>
</div>
- </div>
- </div>
+ )}
+ </ScreenPositionHelper>
<div className="layout-page-main">
<div className="layout-page-main-inner">
diff --git a/server/sonar-web/src/main/js/components/common/ScreenPositionHelper.tsx b/server/sonar-web/src/main/js/components/common/ScreenPositionHelper.tsx
new file mode 100644
index 00000000000..00f8f375766
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/common/ScreenPositionHelper.tsx
@@ -0,0 +1,67 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2017 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 { throttle } from 'lodash';
+
+interface Props {
+ className?: string;
+ children: (position: { top: number; left: number }) => React.ReactElement<any>;
+}
+
+interface State {
+ position: { top: number; left: number };
+}
+
+export default class ScreenPositionHelper extends React.PureComponent<Props, State> {
+ container: HTMLDivElement;
+ throttledUpdatePosition: () => void;
+
+ constructor(props: Props) {
+ super(props);
+ this.state = { position: { top: 0, left: 0 } };
+ this.throttledUpdatePosition = throttle(this.updatePosition, 100);
+ }
+
+ componentDidMount() {
+ this.updatePosition();
+ window.addEventListener('resize', this.throttledUpdatePosition);
+ }
+
+ componentWillUnmount() {
+ window.removeEventListener('resize', this.throttledUpdatePosition);
+ }
+
+ updatePosition = () => {
+ const containerPos = this.container.getBoundingClientRect();
+ this.setState({
+ position: { top: window.scrollY + containerPos.top, left: window.scrollX + containerPos.left }
+ });
+ };
+
+ render() {
+ return (
+ <div
+ className={this.props.className}
+ ref={container => (this.container = container as HTMLDivElement)}>
+ {this.props.children(this.state.position)}
+ </div>
+ );
+ }
+}