aboutsummaryrefslogtreecommitdiffstats
path: root/server
diff options
context:
space:
mode:
authorGrégoire Aubert <gregoire.aubert@sonarsource.com>2018-02-08 15:38:53 +0100
committerGuillaume Jambet <guillaume.jambet@gmail.com>2018-03-01 15:21:05 +0100
commit87ca6669cd77f57d0621c70e53adaa42350ff128 (patch)
tree108e601f1433330f81c319653b84cd9088d12773 /server
parentf44552d26beabb3a99e0483c575bc486a72ef8eb (diff)
downloadsonarqube-87ca6669cd77f57d0621c70e53adaa42350ff128.tar.gz
sonarqube-87ca6669cd77f57d0621c70e53adaa42350ff128.zip
SONAR-10347 Create a BoxedGroupAccordion component and use it in system page
Diffstat (limited to 'server')
-rw-r--r--server/sonar-qa-util/src/main/java/org/sonarqube/qa/util/pageobjects/SystemInfoPage.java20
-rw-r--r--server/sonar-web/src/main/js/app/styles/components/boxed-group.css49
-rw-r--r--server/sonar-web/src/main/js/apps/system/components/ClusterSysInfos.tsx4
-rw-r--r--server/sonar-web/src/main/js/apps/system/components/StandaloneSysInfos.tsx4
-rw-r--r--server/sonar-web/src/main/js/apps/system/components/__tests__/__snapshots__/ClusterSysInfos-test.tsx.snap4
-rw-r--r--server/sonar-web/src/main/js/apps/system/components/__tests__/__snapshots__/StandaloneSysInfos-test.tsx.snap4
-rw-r--r--server/sonar-web/src/main/js/apps/system/components/info-items/HealthCard.tsx80
-rw-r--r--server/sonar-web/src/main/js/apps/system/components/info-items/__tests__/HealthCard-test.tsx22
-rw-r--r--server/sonar-web/src/main/js/apps/system/components/info-items/__tests__/__snapshots__/HealthCard-test.tsx.snap160
-rw-r--r--server/sonar-web/src/main/js/apps/system/styles.css32
-rw-r--r--server/sonar-web/src/main/js/components/controls/BoxedGroupAccordion.tsx78
-rw-r--r--server/sonar-web/src/main/js/components/controls/__tests__/BoxedGroupAccordion-test.tsx52
-rw-r--r--server/sonar-web/src/main/js/components/controls/__tests__/__snapshots__/BoxedGroupAccordion-test.tsx.snap26
13 files changed, 292 insertions, 243 deletions
diff --git a/server/sonar-qa-util/src/main/java/org/sonarqube/qa/util/pageobjects/SystemInfoPage.java b/server/sonar-qa-util/src/main/java/org/sonarqube/qa/util/pageobjects/SystemInfoPage.java
index 9c9802e7e7f..857153eb517 100644
--- a/server/sonar-qa-util/src/main/java/org/sonarqube/qa/util/pageobjects/SystemInfoPage.java
+++ b/server/sonar-qa-util/src/main/java/org/sonarqube/qa/util/pageobjects/SystemInfoPage.java
@@ -21,26 +21,28 @@ package org.sonarqube.qa.util.pageobjects;
import com.codeborne.selenide.CollectionCondition;
import com.codeborne.selenide.Condition;
-import com.codeborne.selenide.Selenide;
+import com.codeborne.selenide.ElementsCollection;
import com.codeborne.selenide.SelenideElement;
+import static com.codeborne.selenide.Selenide.$;
+import static com.codeborne.selenide.Selenide.$$;
+
public class SystemInfoPage {
public SystemInfoPage() {
- Selenide.$(".page-title").should(Condition.exist).shouldHave(Condition.text("System Info"));
- }
-
- public SystemInfoPage shouldHaveCard(String title) {
- Selenide.$$(".system-info-health-card-title").find(Condition.text(title)).should(Condition.exist);
- return this;
+ $(".page-title").should(Condition.exist).shouldHave(Condition.text("System Info"));
}
public SystemInfoPage shouldHaveCards(String... titles) {
- Selenide.$$(".system-info-health-card-title").shouldHave(CollectionCondition.texts(titles));
+ getHealthCards().shouldHave(CollectionCondition.texts(titles));
return this;
}
public SystemInfoPageItem getCardItem(String card) {
- SelenideElement cardTitle = Selenide.$$(".system-info-health-card-title").find(Condition.text(card)).should(Condition.exist);
+ SelenideElement cardTitle = getHealthCards().find(Condition.text(card)).should(Condition.exist);
return new SystemInfoPageItem(cardTitle.parent().parent());
}
+
+ private static ElementsCollection getHealthCards() {
+ return $$(".boxed-group-accordion-title");
+ }
}
diff --git a/server/sonar-web/src/main/js/app/styles/components/boxed-group.css b/server/sonar-web/src/main/js/app/styles/components/boxed-group.css
index 68e97474d38..49aa03123b6 100644
--- a/server/sonar-web/src/main/js/app/styles/components/boxed-group.css
+++ b/server/sonar-web/src/main/js/app/styles/components/boxed-group.css
@@ -26,19 +26,19 @@
.boxed-group > h2 {
line-height: var(--controlHeight);
- padding: 15px 20px 0;
+ padding: calc(2 * var(--gridSize)) 20px 0;
}
.boxed-group hr {
height: 0;
border-top: 1px solid var(--gray94);
- margin: 15px -20px;
+ margin: calc(2 * var(--gridSize)) -20px;
}
.boxed-group-header {
position: relative;
z-index: 10;
- padding: 15px 20px 0;
+ padding: calc(2 * var(--gridSize)) 20px 0;
}
.boxed-group-header > h2 {
@@ -56,12 +56,12 @@
position: relative;
z-index: 12;
float: right;
- margin-top: 15px;
+ margin-top: calc(2 * var(--gridSize));
margin-right: 20px;
}
.boxed-group-inner {
- padding: 15px 20px;
+ padding: calc(2 * var(--gridSize)) 20px;
}
.boxed-group-inner:empty {
@@ -69,12 +69,45 @@
}
.boxed-group-list {
- margin-top: -8px;
- margin-bottom: -8px;
+ margin-top: - var(--gridSize);
+ margin-bottom: - var(--gridSize);
}
.boxed-group-list > li {
margin-left: -20px;
margin-right: -20px;
- padding: 8px 20px;
+ padding: var(--gridSize) 20px;
+}
+
+.boxed-group-accordion {
+ margin-bottom: var(--gridSize);
+ transition: border-color 0.3s ease;
+}
+
+.boxed-group-accordion:not(.no-hover):hover {
+ border-color: var(--blue);
+}
+
+.boxed-group-accordion:not(.no-hover):hover .boxed-group-accordion-title {
+ color: var(--blue);
+}
+
+.boxed-group-accordion .boxed-group-header {
+ cursor: pointer;
+ padding-bottom: calc(2 * var(--gridSize));
+}
+
+.boxed-group-accordion .boxed-group-header > .alert {
+ display: inline-block;
+ margin-bottom: -6px;
+ margin-top: -6px;
+}
+
+.boxed-group-accordion .boxed-group-inner {
+ padding-top: 0;
+}
+
+.boxed-group-accordion-title {
+ font-weight: bold;
+ transition: color 0.3s ease;
}
diff --git a/server/sonar-web/src/main/js/apps/system/components/ClusterSysInfos.tsx b/server/sonar-web/src/main/js/apps/system/components/ClusterSysInfos.tsx
index dec9d88310f..1d7ae6d3f8c 100644
--- a/server/sonar-web/src/main/js/apps/system/components/ClusterSysInfos.tsx
+++ b/server/sonar-web/src/main/js/apps/system/components/ClusterSysInfos.tsx
@@ -41,7 +41,7 @@ interface Props {
export default function ClusterSysInfos({ expandedCards, sysInfoData, toggleCard }: Props) {
const mainCardName = 'System';
return (
- <ul>
+ <>
<HealthCard
biggerHealth={true}
health={getHealth(sysInfoData)}
@@ -77,6 +77,6 @@ export default function ClusterSysInfos({ expandedCards, sysInfoData, toggleCard
sysInfoData={ignoreInfoFields(node)}
/>
))}
- </ul>
+ </>
);
}
diff --git a/server/sonar-web/src/main/js/apps/system/components/StandaloneSysInfos.tsx b/server/sonar-web/src/main/js/apps/system/components/StandaloneSysInfos.tsx
index ba09a11934a..ecdb45fe85c 100644
--- a/server/sonar-web/src/main/js/apps/system/components/StandaloneSysInfos.tsx
+++ b/server/sonar-web/src/main/js/apps/system/components/StandaloneSysInfos.tsx
@@ -38,7 +38,7 @@ interface Props {
export default function StandAloneSysInfos({ expandedCards, sysInfoData, toggleCard }: Props) {
const mainCardName = 'System';
return (
- <ul>
+ <>
<HealthCard
biggerHealth={true}
health={getHealth(sysInfoData)}
@@ -57,6 +57,6 @@ export default function StandAloneSysInfos({ expandedCards, sysInfoData, toggleC
sysInfoData={ignoreInfoFields(section)}
/>
))}
- </ul>
+ </>
);
}
diff --git a/server/sonar-web/src/main/js/apps/system/components/__tests__/__snapshots__/ClusterSysInfos-test.tsx.snap b/server/sonar-web/src/main/js/apps/system/components/__tests__/__snapshots__/ClusterSysInfos-test.tsx.snap
index ca216f72e04..0e31e22e4be 100644
--- a/server/sonar-web/src/main/js/apps/system/components/__tests__/__snapshots__/ClusterSysInfos-test.tsx.snap
+++ b/server/sonar-web/src/main/js/apps/system/components/__tests__/__snapshots__/ClusterSysInfos-test.tsx.snap
@@ -1,7 +1,7 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`should support more than two nodes 1`] = `
-<ul>
+<React.Fragment>
<HealthCard
biggerHealth={true}
health="RED"
@@ -67,5 +67,5 @@ exports[`should support more than two nodes 1`] = `
}
}
/>
-</ul>
+</React.Fragment>
`;
diff --git a/server/sonar-web/src/main/js/apps/system/components/__tests__/__snapshots__/StandaloneSysInfos-test.tsx.snap b/server/sonar-web/src/main/js/apps/system/components/__tests__/__snapshots__/StandaloneSysInfos-test.tsx.snap
index e5832de513c..0d7912636bf 100644
--- a/server/sonar-web/src/main/js/apps/system/components/__tests__/__snapshots__/StandaloneSysInfos-test.tsx.snap
+++ b/server/sonar-web/src/main/js/apps/system/components/__tests__/__snapshots__/StandaloneSysInfos-test.tsx.snap
@@ -1,7 +1,7 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`should render correctly 1`] = `
-<ul>
+<React.Fragment>
<HealthCard
biggerHealth={true}
health="RED"
@@ -60,5 +60,5 @@ exports[`should render correctly 1`] = `
}
}
/>
-</ul>
+</React.Fragment>
`;
diff --git a/server/sonar-web/src/main/js/apps/system/components/info-items/HealthCard.tsx b/server/sonar-web/src/main/js/apps/system/components/info-items/HealthCard.tsx
index 105c8002bcd..f042eed7de3 100644
--- a/server/sonar-web/src/main/js/apps/system/components/info-items/HealthCard.tsx
+++ b/server/sonar-web/src/main/js/apps/system/components/info-items/HealthCard.tsx
@@ -18,11 +18,10 @@
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
import * as React from 'react';
-import * as classNames from 'classnames';
import { map } from 'lodash';
import HealthItem from './HealthItem';
import Section from './Section';
-import OpenCloseIcon from '../../../../components/icons-components/OpenCloseIcon';
+import BoxedGroupAccordion from '../../../../components/controls/BoxedGroupAccordion';
import { HealthType, SysValueObject } from '../../../../api/system';
import { LOGS_LEVELS, groupSections, getLogsLevel } from '../../utils';
import { translate } from '../../../../helpers/l10n';
@@ -37,34 +36,27 @@ interface Props {
sysInfoData: SysValueObject;
}
-interface State {
- hoveringDetail: boolean;
-}
-
-export default class HealthCard extends React.PureComponent<Props, State> {
- state: State = { hoveringDetail: false };
-
- handleClick = () => this.props.onClick(this.props.name);
- onDetailEnter = () => this.setState({ hoveringDetail: true });
- onDetailLeave = () => this.setState({ hoveringDetail: false });
-
- render() {
- const { health, open, sysInfoData } = this.props;
- const { mainSection, sections } = groupSections(sysInfoData);
- const showFields = open && mainSection && Object.keys(mainSection).length > 0;
- const showSections = open && sections;
- const logLevel = getLogsLevel(sysInfoData);
- const showLogLevelWarning = logLevel && logLevel !== LOGS_LEVELS[0];
- return (
- <li
- className={classNames('boxed-group system-info-health-card', {
- 'no-hover': this.state.hoveringDetail
- })}>
- <div className="boxed-group-header" onClick={this.handleClick}>
- <span className="system-info-health-card-title">
- <OpenCloseIcon className="little-spacer-right" open={open} />
- {this.props.name}
- </span>
+export default function HealthCard({
+ biggerHealth,
+ health,
+ healthCauses,
+ onClick,
+ open,
+ name,
+ sysInfoData
+}: Props) {
+ const { mainSection, sections } = groupSections(sysInfoData);
+ const showFields = open && mainSection && Object.keys(mainSection).length > 0;
+ const showSections = open && sections;
+ const logLevel = getLogsLevel(sysInfoData);
+ const showLogLevelWarning = logLevel && logLevel !== LOGS_LEVELS[0];
+ return (
+ <BoxedGroupAccordion
+ data={name}
+ onClick={onClick}
+ open={open}
+ renderHeader={() => (
+ <>
{showLogLevelWarning && (
<span className="alert alert-danger spacer-left">
{translate('system.log_level.warning.short')}
@@ -72,25 +64,19 @@ export default class HealthCard extends React.PureComponent<Props, State> {
)}
{health && (
<HealthItem
- biggerHealth={this.props.biggerHealth}
+ biggerHealth={biggerHealth}
className="pull-right"
health={health}
- healthCauses={this.props.healthCauses}
- name={this.props.name}
+ healthCauses={healthCauses}
+ name={name}
/>
)}
- </div>
- {open && (
- <div
- className="boxed-group-inner"
- onMouseEnter={this.onDetailEnter}
- onMouseLeave={this.onDetailLeave}>
- {showFields && <Section items={mainSection} />}
- {showSections &&
- map(sections, (section, name) => <Section key={name} items={section} name={name} />)}
- </div>
- )}
- </li>
- );
- }
+ </>
+ )}
+ title={name}>
+ {showFields && <Section items={mainSection} />}
+ {showSections &&
+ map(sections, (section, name) => <Section key={name} items={section} name={name} />)}
+ </BoxedGroupAccordion>
+ );
}
diff --git a/server/sonar-web/src/main/js/apps/system/components/info-items/__tests__/HealthCard-test.tsx b/server/sonar-web/src/main/js/apps/system/components/info-items/__tests__/HealthCard-test.tsx
index abe3f8b002b..08ea6b5f66d 100644
--- a/server/sonar-web/src/main/js/apps/system/components/info-items/__tests__/HealthCard-test.tsx
+++ b/server/sonar-web/src/main/js/apps/system/components/info-items/__tests__/HealthCard-test.tsx
@@ -20,22 +20,10 @@
import * as React from 'react';
import { shallow } from 'enzyme';
import HealthCard from '../HealthCard';
-import { click } from '../../../../../helpers/testUtils';
import { HealthType } from '../../../../../api/system';
it('should render correctly', () => {
- expect(getShallowWrapper()).toMatchSnapshot();
-});
-
-it('should display the sysinfo detail', () => {
- expect(getShallowWrapper({ biggerHealth: true, open: true })).toMatchSnapshot();
-});
-
-it('should show the sysinfo detail when the card is clicked', () => {
- const onClick = jest.fn();
- click(getShallowWrapper({ onClick }).find('.boxed-group-header'));
- expect(onClick).toBeCalled();
- expect(onClick).toBeCalledWith('Foobar');
+ expect(getWrapper()).toMatchSnapshot();
});
it('should show a main section and multiple sub sections', () => {
@@ -45,16 +33,18 @@ it('should show a main section and multiple sub sections', () => {
Database: { db: 'test' },
Elasticseach: { Elastic: 'search' }
};
- expect(getShallowWrapper({ open: true, sysInfoData })).toMatchSnapshot();
+ expect(getWrapper({ open: true, sysInfoData })).toMatchSnapshot();
});
it('should display the log level alert', () => {
expect(
- getShallowWrapper({ sysInfoData: { 'Logs Level': 'DEBUG' } }).find('.alert')
+ getWrapper({ sysInfoData: { 'Logs Level': 'DEBUG' } })
+ .dive()
+ .find('.alert')
).toMatchSnapshot();
});
-function getShallowWrapper(props = {}) {
+function getWrapper(props = {}) {
return shallow(
<HealthCard
biggerHealth={false}
diff --git a/server/sonar-web/src/main/js/apps/system/components/info-items/__tests__/__snapshots__/HealthCard-test.tsx.snap b/server/sonar-web/src/main/js/apps/system/components/info-items/__tests__/__snapshots__/HealthCard-test.tsx.snap
index fe7c59a4286..8ce1a5d4f14 100644
--- a/server/sonar-web/src/main/js/apps/system/components/info-items/__tests__/__snapshots__/HealthCard-test.tsx.snap
+++ b/server/sonar-web/src/main/js/apps/system/components/info-items/__tests__/__snapshots__/HealthCard-test.tsx.snap
@@ -8,135 +8,49 @@ exports[`should display the log level alert 1`] = `
</span>
`;
-exports[`should display the sysinfo detail 1`] = `
-<li
- className="boxed-group system-info-health-card"
->
- <div
- className="boxed-group-header"
- onClick={[Function]}
- >
- <span
- className="system-info-health-card-title"
- >
- <OpenCloseIcon
- className="little-spacer-right"
- open={true}
- />
- Foobar
- </span>
- <HealthItem
- biggerHealth={true}
- className="pull-right"
- health="RED"
- healthCauses={
- Array [
- "foo",
- ]
- }
- name="Foobar"
- />
- </div>
- <div
- className="boxed-group-inner"
- onMouseEnter={[Function]}
- onMouseLeave={[Function]}
- />
-</li>
-`;
-
exports[`should render correctly 1`] = `
-<li
- className="boxed-group system-info-health-card"
->
- <div
- className="boxed-group-header"
- onClick={[Function]}
- >
- <span
- className="system-info-health-card-title"
- >
- <OpenCloseIcon
- className="little-spacer-right"
- open={false}
- />
- Foobar
- </span>
- <HealthItem
- biggerHealth={false}
- className="pull-right"
- health="RED"
- healthCauses={
- Array [
- "foo",
- ]
- }
- name="Foobar"
- />
- </div>
-</li>
+<BoxedGroupAccordion
+ data="Foobar"
+ onClick={[Function]}
+ open={false}
+ renderHeader={[Function]}
+ title="Foobar"
+/>
`;
exports[`should show a main section and multiple sub sections 1`] = `
-<li
- className="boxed-group system-info-health-card"
+<BoxedGroupAccordion
+ data="Foobar"
+ onClick={[Function]}
+ open={true}
+ renderHeader={[Function]}
+ title="Foobar"
>
- <div
- className="boxed-group-header"
- onClick={[Function]}
- >
- <span
- className="system-info-health-card-title"
- >
- <OpenCloseIcon
- className="little-spacer-right"
- open={true}
- />
- Foobar
- </span>
- <HealthItem
- biggerHealth={false}
- className="pull-right"
- health="RED"
- healthCauses={
- Array [
- "foo",
- ]
- }
- name="Foobar"
- />
- </div>
- <div
- className="boxed-group-inner"
- onMouseEnter={[Function]}
- onMouseLeave={[Function]}
- >
- <Section
- items={
- Object {
- "Name": "foo",
- "bar": "Bar",
- }
+ <Section
+ items={
+ Object {
+ "Name": "foo",
+ "bar": "Bar",
}
- />
- <Section
- items={
- Object {
- "db": "test",
- }
+ }
+ />
+ <Section
+ items={
+ Object {
+ "db": "test",
}
- key="Database"
- name="Database"
- />
- <Section
- items={
- Object {
- "Elastic": "search",
- }
+ }
+ key="Database"
+ name="Database"
+ />
+ <Section
+ items={
+ Object {
+ "Elastic": "search",
}
- key="Elasticseach"
- name="Elasticseach"
- />
- </div>
-</li>
+ }
+ key="Elasticseach"
+ name="Elasticseach"
+ />
+</BoxedGroupAccordion>
`;
diff --git a/server/sonar-web/src/main/js/apps/system/styles.css b/server/sonar-web/src/main/js/apps/system/styles.css
index c6c4ad4afb5..3c332eaa7c1 100644
--- a/server/sonar-web/src/main/js/apps/system/styles.css
+++ b/server/sonar-web/src/main/js/apps/system/styles.css
@@ -22,38 +22,6 @@
margin-bottom: 16px;
}
-.system-info-health-card {
- margin-bottom: 8px;
- transition: border-color 0.3s ease;
-}
-
-.system-info-health-card:not(.no-hover):hover {
- border-color: var(--blue);
-}
-
-.system-info-health-card:not(.no-hover):hover .system-info-health-card-title {
- color: var(--blue);
-}
-
-.system-info-health-card .boxed-group-header {
- cursor: pointer;
- padding-bottom: 15px;
-}
-
-.system-info-health-card .boxed-group-header > .alert {
- display: inline-block;
- margin-bottom: -6px;
- margin-top: -6px;
-}
-
-.system-info-health-card .boxed-group-inner {
- padding-top: 0;
-}
-
-.system-info-health-card-title {
- font-weight: bold;
-}
-
.system-info-health-info {
margin-top: -12px;
}
diff --git a/server/sonar-web/src/main/js/components/controls/BoxedGroupAccordion.tsx b/server/sonar-web/src/main/js/components/controls/BoxedGroupAccordion.tsx
new file mode 100644
index 00000000000..63129cc5b7f
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/controls/BoxedGroupAccordion.tsx
@@ -0,0 +1,78 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2018 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 classNames from 'classnames';
+import OpenCloseIcon from '../icons-components/OpenCloseIcon';
+
+interface Props {
+ children: React.ReactNode;
+ className?: string;
+ data?: string;
+ onClick: (data?: string) => void;
+ open: boolean;
+ renderHeader?: () => React.ReactNode;
+ title: React.ReactNode;
+}
+
+interface State {
+ hoveringInner: boolean;
+}
+
+export default class BoxedGroupAccordion extends React.PureComponent<Props, State> {
+ state: State = { hoveringInner: false };
+
+ handleClick = () => {
+ this.props.onClick(this.props.data);
+ };
+
+ onDetailEnter = () => {
+ this.setState({ hoveringInner: true });
+ };
+
+ onDetailLeave = () => {
+ this.setState({ hoveringInner: false });
+ };
+
+ render() {
+ const { className, open, renderHeader, title } = this.props;
+ return (
+ <div
+ className={classNames('boxed-group boxed-group-accordion', className, {
+ 'no-hover': this.state.hoveringInner
+ })}>
+ <div className="boxed-group-header" onClick={this.handleClick} role="listitem">
+ <span className="boxed-group-accordion-title">
+ <OpenCloseIcon className="little-spacer-right" open={open} />
+ {title}
+ </span>
+ {renderHeader && renderHeader()}
+ </div>
+ {open && (
+ <div
+ className="boxed-group-inner"
+ onMouseEnter={this.onDetailEnter}
+ onMouseLeave={this.onDetailLeave}>
+ {this.props.children}
+ </div>
+ )}
+ </div>
+ );
+ }
+}
diff --git a/server/sonar-web/src/main/js/components/controls/__tests__/BoxedGroupAccordion-test.tsx b/server/sonar-web/src/main/js/components/controls/__tests__/BoxedGroupAccordion-test.tsx
new file mode 100644
index 00000000000..8101724be4b
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/controls/__tests__/BoxedGroupAccordion-test.tsx
@@ -0,0 +1,52 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2018 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 { shallow } from 'enzyme';
+import { click } from '../../../helpers/testUtils';
+import BoxedGroupAccordion from '../BoxedGroupAccordion';
+
+it('should render correctly', () => {
+ expect(getWrapper()).toMatchSnapshot();
+});
+
+it('should show the inner content after a click', () => {
+ const onClick = jest.fn();
+ const wrapper = getWrapper({ onClick });
+ click(wrapper.find('.boxed-group-header'));
+
+ expect(onClick).lastCalledWith('foo');
+ wrapper.setProps({ open: true });
+
+ expect(wrapper.find('.boxed-group-inner').exists()).toBeTruthy();
+});
+
+function getWrapper(props = {}) {
+ return shallow(
+ <BoxedGroupAccordion
+ data="foo"
+ onClick={() => {}}
+ open={false}
+ renderHeader={() => <div>header content</div>}
+ title="Foo"
+ {...props}>
+ <div>inner content</div>
+ </BoxedGroupAccordion>
+ );
+}
diff --git a/server/sonar-web/src/main/js/components/controls/__tests__/__snapshots__/BoxedGroupAccordion-test.tsx.snap b/server/sonar-web/src/main/js/components/controls/__tests__/__snapshots__/BoxedGroupAccordion-test.tsx.snap
new file mode 100644
index 00000000000..0c7b74b79d9
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/controls/__tests__/__snapshots__/BoxedGroupAccordion-test.tsx.snap
@@ -0,0 +1,26 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`should render correctly 1`] = `
+<div
+ className="boxed-group boxed-group-accordion"
+>
+ <div
+ className="boxed-group-header"
+ onClick={[Function]}
+ role="listitem"
+ >
+ <span
+ className="boxed-group-accordion-title"
+ >
+ <OpenCloseIcon
+ className="little-spacer-right"
+ open={false}
+ />
+ Foo
+ </span>
+ <div>
+ header content
+ </div>
+ </div>
+</div>
+`;