aboutsummaryrefslogtreecommitdiffstats
path: root/server/sonar-web/src/main/js/components
diff options
context:
space:
mode:
authorStas Vilchik <stas.vilchik@sonarsource.com>2018-02-21 13:32:25 +0100
committerGitHub <noreply@github.com>2018-02-21 13:32:25 +0100
commitd6d2b5824be0c018ad4fc606f01abfb9f266c804 (patch)
treef6e8fde0634860a56c9f6d43107187cf21a0b2f3 /server/sonar-web/src/main/js/components
parentcaf48455f693b76d06ebd2328d8498442cb0c1ab (diff)
downloadsonarqube-d6d2b5824be0c018ad4fc606f01abfb9f266c804.tar.gz
sonarqube-d6d2b5824be0c018ad4fc606f01abfb9f266c804.zip
review source viewer measures overlay in react (#3084)
Diffstat (limited to 'server/sonar-web/src/main/js/components')
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/SourceViewerBase.js16
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/SourceViewerHeader.tsx (renamed from server/sonar-web/src/main/js/components/SourceViewer/SourceViewerHeader.js)90
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlay.tsx459
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayCoveredFiles.tsx115
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayMeasure.tsx57
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayTestCase.tsx62
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayTestCases.tsx181
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlay-test.tsx172
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayCoveredFiles-test.tsx55
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayMeasure-test.tsx48
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayTestCase-test.tsx42
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayTestCases-test.tsx74
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlay-test.tsx.snap1535
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayCoveredFiles-test.tsx.snap76
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayMeasure-test.tsx.snap53
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayTestCase-test.tsx.snap34
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayTestCases-test.tsx.snap247
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/styles.css1
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/views/measures-overlay.js282
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-all.hbs42
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-coverage.hbs36
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-duplications.hbs30
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-issues.hbs47
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-lines.hbs29
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-test-cases.hbs72
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-tests.hbs40
-rw-r--r--server/sonar-web/src/main/js/components/SourceViewer/views/templates/source-viewer-measures.hbs68
-rw-r--r--server/sonar-web/src/main/js/components/measure/Measure.tsx16
-rw-r--r--server/sonar-web/src/main/js/components/measure/__tests__/Measure-test.tsx4
-rw-r--r--server/sonar-web/src/main/js/components/shared/TestStatusIcon.tsx30
30 files changed, 3303 insertions, 710 deletions
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/SourceViewerBase.js b/server/sonar-web/src/main/js/components/SourceViewer/SourceViewerBase.js
index 2022165e43f..6f267452fa0 100644
--- a/server/sonar-web/src/main/js/components/SourceViewer/SourceViewerBase.js
+++ b/server/sonar-web/src/main/js/components/SourceViewer/SourceViewerBase.js
@@ -27,7 +27,6 @@ import CoveragePopupView from './popups/coverage-popup';
import DuplicationPopupView from './popups/duplication-popup';
import LineActionsPopupView from './popups/line-actions-popup';
import SCMPopupView from './popups/scm-popup';
-import MeasuresOverlay from './views/measures-overlay';
import loadIssues from './helpers/loadIssues';
import getCoverageStatus from './helpers/getCoverageStatus';
import {
@@ -464,15 +463,6 @@ export default class SourceViewerBase extends React.PureComponent {
});
};
- showMeasures = () => {
- const measuresOverlay = new MeasuresOverlay({
- branch: this.props.branch,
- component: this.state.component,
- large: true
- });
- measuresOverlay.render();
- };
-
handleCoverageClick = (line /*: SourceLine */, element /*: HTMLElement */) => {
getTests(this.props.component, line.line, this.props.branch).then(tests => {
const popup = new CoveragePopupView({
@@ -691,11 +681,7 @@ export default class SourceViewerBase extends React.PureComponent {
return (
<div className={className} ref={node => (this.node = node)}>
- <SourceViewerHeader
- branch={this.props.branch}
- component={this.state.component}
- showMeasures={this.showMeasures}
- />
+ <SourceViewerHeader branch={this.props.branch} sourceViewerFile={this.state.component} />
{sourceRemoved && (
<div className="alert alert-warning spacer-top">
{translate('code_viewer.no_source_code_displayed_due_to_source_removed')}
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/SourceViewerHeader.js b/server/sonar-web/src/main/js/components/SourceViewer/SourceViewerHeader.tsx
index 253f8db02ac..a29348b9312 100644
--- a/server/sonar-web/src/main/js/components/SourceViewer/SourceViewerHeader.js
+++ b/server/sonar-web/src/main/js/components/SourceViewer/SourceViewerHeader.tsx
@@ -17,49 +17,46 @@
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
-// @flow
-import React from 'react';
+import * as React from 'react';
import { Link } from 'react-router';
+import MeasuresOverlay from './components/MeasuresOverlay';
+import { SourceViewerFile } from '../../app/types';
import QualifierIcon from '../shared/QualifierIcon';
import FavoriteContainer from '../controls/FavoriteContainer';
-import { getPathUrlAsString, getProjectUrl, getComponentIssuesUrl } from '../../helpers/urls';
+import {
+ getPathUrlAsString,
+ getProjectUrl,
+ getComponentIssuesUrl,
+ getBaseUrl
+} from '../../helpers/urls';
import { collapsedDirFromPath, fileFromPath } from '../../helpers/path';
import { translate } from '../../helpers/l10n';
import { formatMeasure } from '../../helpers/measures';
-export default class SourceViewerHeader extends React.PureComponent {
- /*:: props: {
- branch?: string,
- component: {
- canMarkAsFavorite: boolean,
- key: string,
- measures: {
- coverage?: string,
- duplicationDensity?: string,
- issues?: string,
- lines?: string,
- tests?: string
- },
- path: string,
- project: string,
- projectName: string,
- q: string,
- subProject?: string,
- subProjectName?: string,
- uuid: string
- },
- showMeasures: () => void
+interface Props {
+ branch: string | undefined;
+ sourceViewerFile: SourceViewerFile;
+}
+
+interface State {
+ measuresOverlay: boolean;
+}
+
+export default class SourceViewerHeader extends React.PureComponent<Props, State> {
+ state: State = { measuresOverlay: false };
+
+ handleShowMeasuresClick = (event: React.SyntheticEvent<HTMLAnchorElement>) => {
+ event.preventDefault();
+ this.setState({ measuresOverlay: true });
};
-*/
- showMeasures = (e /*: SyntheticInputEvent */) => {
- e.preventDefault();
- this.props.showMeasures();
+ handleMeasuresOverlayClose = () => {
+ this.setState({ measuresOverlay: false });
};
- openInWorkspace = (e /*: SyntheticInputEvent */) => {
- e.preventDefault();
- const { key } = this.props.component;
+ openInWorkspace = (event: React.SyntheticEvent<HTMLAnchorElement>) => {
+ event.preventDefault();
+ const { key } = this.props.sourceViewerFile;
const Workspace = require('../workspace/main').default;
Workspace.openComponent({ key, branch: this.props.branch });
};
@@ -75,11 +72,11 @@ export default class SourceViewerHeader extends React.PureComponent {
subProject,
subProjectName,
uuid
- } = this.props.component;
+ } = this.props.sourceViewerFile;
const isUnitTest = q === 'UTS';
const workspace = false;
let rawSourcesLink =
- window.baseUrl + `/api/sources/raw?key=${encodeURIComponent(this.props.component.key)}`;
+ getBaseUrl() + `/api/sources/raw?key=${encodeURIComponent(this.props.sourceViewerFile.key)}`;
if (this.props.branch) {
rawSourcesLink += `&branch=${encodeURIComponent(this.props.branch)}`;
}
@@ -91,8 +88,8 @@ export default class SourceViewerHeader extends React.PureComponent {
<div className="component-name">
<div className="component-name-parent">
<a
- href={getPathUrlAsString(getProjectUrl(project, this.props.branch))}
- className="link-with-icon">
+ className="link-with-icon"
+ href={getPathUrlAsString(getProjectUrl(project, this.props.branch))}>
<QualifierIcon qualifier="TRK" /> <span>{projectName}</span>
</a>
</div>
@@ -100,8 +97,8 @@ export default class SourceViewerHeader extends React.PureComponent {
{subProject != null && (
<div className="component-name-parent">
<a
- href={getPathUrlAsString(getProjectUrl(subProject, this.props.branch))}
- className="link-with-icon">
+ className="link-with-icon"
+ href={getPathUrlAsString(getProjectUrl(subProject, this.props.branch))}>
<QualifierIcon qualifier="BRC" /> <span>{subProjectName}</span>
</a>
</div>
@@ -110,7 +107,7 @@ export default class SourceViewerHeader extends React.PureComponent {
<div className="component-name-path">
<QualifierIcon qualifier={q} /> <span>{collapsedDirFromPath(path)}</span>
<span className="component-name-file">{fileFromPath(path)}</span>
- {this.props.component.canMarkAsFavorite && (
+ {this.props.sourceViewerFile.canMarkAsFavorite && (
<FavoriteContainer className="component-name-favorite" componentKey={key} />
)}
</div>
@@ -125,18 +122,25 @@ export default class SourceViewerHeader extends React.PureComponent {
/>
<ul className="dropdown-menu dropdown-menu-right">
<li>
- <a className="js-measures" href="#" onClick={this.showMeasures}>
+ <a className="js-measures" href="#" onClick={this.handleShowMeasuresClick}>
{translate('component_viewer.show_details')}
</a>
+ {this.state.measuresOverlay && (
+ <MeasuresOverlay
+ branch={this.props.branch}
+ onClose={this.handleMeasuresOverlayClose}
+ sourceViewerFile={this.props.sourceViewerFile}
+ />
+ )}
</li>
<li>
<a
className="js-new-window"
- target="_blank"
href={getPathUrlAsString({
pathname: '/component',
- query: { branch: this.props.branch, id: this.props.component.key }
- })}>
+ query: { branch: this.props.branch, id: this.props.sourceViewerFile.key }
+ })}
+ target="_blank">
{translate('component_viewer.new_window')}
</a>
</li>
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlay.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlay.tsx
new file mode 100644
index 00000000000..131c58b1a4f
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlay.tsx
@@ -0,0 +1,459 @@
+/*
+ * 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 { Link } from 'react-router';
+import { keyBy, sortBy, groupBy } from 'lodash';
+import MeasuresOverlayMeasure from './MeasuresOverlayMeasure';
+import MeasuresOverlayTestCases from './MeasuresOverlayTestCases';
+import { getFacets } from '../../../api/issues';
+import { getMeasures } from '../../../api/measures';
+import { getAllMetrics } from '../../../api/metrics';
+import { FacetValue, SourceViewerFile } from '../../../app/types';
+import Modal from '../../controls/Modal';
+import Measure from '../../measure/Measure';
+import QualifierIcon from '../../shared/QualifierIcon';
+import SeverityHelper from '../../shared/SeverityHelper';
+import CoverageRating from '../../ui/CoverageRating';
+import DuplicationsRating from '../../ui/DuplicationsRating';
+import IssueTypeIcon from '../../ui/IssueTypeIcon';
+import { SEVERITIES, TYPES } from '../../../helpers/constants';
+import { translate, getLocalizedMetricName } from '../../../helpers/l10n';
+import {
+ formatMeasure,
+ MeasureEnhanced,
+ getDisplayMetrics,
+ enhanceMeasuresWithMetrics
+} from '../../../helpers/measures';
+import { getProjectUrl } from '../../../helpers/urls';
+
+interface Props {
+ branch: string | undefined;
+ onClose: () => void;
+ sourceViewerFile: SourceViewerFile;
+}
+
+interface Measures {
+ [metricKey: string]: MeasureEnhanced;
+}
+
+interface State {
+ loading: boolean;
+ measures: Measures;
+ severitiesFacet?: FacetValue[];
+ showAllMeasures: boolean;
+ tagsFacet?: FacetValue[];
+ typesFacet?: FacetValue[];
+}
+
+export default class MeasuresOverlay extends React.PureComponent<Props, State> {
+ mounted = false;
+ state: State = { loading: true, measures: {}, showAllMeasures: false };
+
+ componentDidMount() {
+ this.mounted = true;
+ this.fetchData();
+ }
+
+ componentWillUnmount() {
+ this.mounted = false;
+ }
+
+ fetchData = () => {
+ Promise.all([this.fetchMeasures(), this.fetchIssues()]).then(
+ ([measures, facets]) => {
+ if (this.mounted) {
+ this.setState({ loading: false, measures, ...facets });
+ }
+ },
+ () => {
+ if (this.mounted) {
+ this.setState({ loading: false });
+ }
+ }
+ );
+ };
+
+ fetchMeasures = () => {
+ return getAllMetrics().then(metrics => {
+ const metricKeys = getDisplayMetrics(metrics).map(metric => metric.key);
+
+ // eslint-disable-next-line promise/no-nesting
+ return getMeasures(this.props.sourceViewerFile.key, metricKeys, this.props.branch).then(
+ measures => {
+ const withMetrics = enhanceMeasuresWithMetrics(measures, metrics).filter(
+ measure => measure.metric
+ );
+ return keyBy(withMetrics, measure => measure.metric.key);
+ }
+ );
+ });
+ };
+
+ fetchIssues = () => {
+ return getFacets(
+ {
+ branch: this.props.branch,
+ componentKeys: this.props.sourceViewerFile.key,
+ resolved: 'false'
+ },
+ ['types', 'severities', 'tags']
+ ).then(({ facets }) => {
+ const severitiesFacet = facets.find(f => f.property === 'severities');
+ const tagsFacet = facets.find(f => f.property === 'tags');
+ const typesFacet = facets.find(f => f.property === 'types');
+ return {
+ severitiesFacet: severitiesFacet && severitiesFacet.values,
+ tagsFacet: tagsFacet && tagsFacet.values,
+ typesFacet: typesFacet && typesFacet.values
+ };
+ });
+ };
+
+ handleCloseClick = (event: React.SyntheticEvent<HTMLButtonElement>) => {
+ event.preventDefault();
+ event.currentTarget.blur();
+ this.props.onClose();
+ };
+
+ handleAllMeasuresClick = (event: React.SyntheticEvent<HTMLAnchorElement>) => {
+ event.preventDefault();
+ event.currentTarget.blur();
+ this.setState({ showAllMeasures: true });
+ };
+
+ renderMeasure = (measure: MeasureEnhanced | undefined) => {
+ return measure ? <MeasuresOverlayMeasure key={measure.metric.key} measure={measure} /> : null;
+ };
+
+ renderLines = () => {
+ const { measures } = this.state;
+
+ return (
+ <div className="source-viewer-measures-section">
+ <div className="source-viewer-measures-card">
+ <div className="measures">
+ <div className="measures-list">
+ {this.renderMeasure(measures.lines)}
+ {this.renderMeasure(measures.ncloc)}
+ {this.renderMeasure(measures.comment_lines)}
+ {this.renderMeasure(measures.comment_lines_density)}
+ </div>
+ </div>
+
+ <div className="measures">
+ <div className="measures-list">
+ {this.renderMeasure(measures.cognitive_complexity)}
+ {this.renderMeasure(measures.complexity)}
+ {this.renderMeasure(measures.function_complexity)}
+ </div>
+ </div>
+ </div>
+ </div>
+ );
+ };
+
+ renderBigMeasure = (measure: MeasureEnhanced | undefined) => {
+ return measure ? (
+ <div className="measure measure-big" data-metric={measure.metric.key}>
+ <span className="measure-value">
+ <Measure
+ metricKey={measure.metric.key}
+ metricType={measure.metric.type}
+ value={measure.value}
+ />
+ </span>
+ <span className="measure-name">{getLocalizedMetricName(measure.metric, true)}</span>
+ </div>
+ ) : null;
+ };
+
+ renderIssues = () => {
+ const { measures, severitiesFacet, tagsFacet, typesFacet } = this.state;
+ return (
+ <div className="source-viewer-measures-section">
+ <div className="source-viewer-measures-card">
+ <div className="measures">
+ {this.renderBigMeasure(measures.violations)}
+ {this.renderBigMeasure(measures.sqale_index)}
+ </div>
+ {measures.violations &&
+ !measures.violations.value && (
+ <>
+ {typesFacet && (
+ <div className="measures">
+ <div className="measures-list">
+ {sortBy(typesFacet, f => TYPES.indexOf(f.val)).map(f => (
+ <div className="measure measure-one-line" key={f.val}>
+ <span className="measure-name">
+ <IssueTypeIcon className="little-spacer-right" query={f.val} />
+ {translate('issue.type', f.val)}
+ </span>
+ <span className="measure-value">
+ {formatMeasure(f.count, 'SHORT_INT')}
+ </span>
+ </div>
+ ))}
+ </div>
+ </div>
+ )}
+ {severitiesFacet && (
+ <div className="measures">
+ <div className="measures-list">
+ {sortBy(severitiesFacet, f => SEVERITIES.indexOf(f.val)).map(f => (
+ <div className="measure measure-one-line" key={f.val}>
+ <span className="measure-name">
+ <SeverityHelper severity={f.val} />
+ </span>
+ <span className="measure-value">
+ {formatMeasure(f.count, 'SHORT_INT')}
+ </span>
+ </div>
+ ))}
+ </div>
+ </div>
+ )}
+ {tagsFacet && (
+ <div className="measures">
+ <div className="measures-list">
+ {tagsFacet.map(f => (
+ <div className="measure measure-one-line" key={f.val}>
+ <span className="measure-name">
+ <i className="icon-tags little-spacer-right" />
+ {f.val}
+ </span>
+ <span className="measure-value">
+ {formatMeasure(f.count, 'SHORT_INT')}
+ </span>
+ </div>
+ ))}
+ </div>
+ </div>
+ )}
+ </>
+ )}
+ </div>
+ </div>
+ );
+ };
+
+ renderCoverage = () => {
+ const { coverage } = this.state.measures;
+ if (!coverage) {
+ return null;
+ }
+ return (
+ <div className="source-viewer-measures-section">
+ <div className="source-viewer-measures-card">
+ <div className="measures">
+ <div className="measures-chart">
+ <CoverageRating size="big" value={coverage.value} />
+ </div>
+ <div className="measure measure-big" data-metric={coverage.metric.key}>
+ <span className="measure-value">
+ <Measure
+ metricKey={coverage.metric.key}
+ metricType={coverage.metric.type}
+ value={coverage.value}
+ />
+ </span>
+ <span className="measure-name">{getLocalizedMetricName(coverage.metric)}</span>
+ </div>
+ </div>
+
+ <div className="measures">
+ <div className="measures-list">
+ {this.renderMeasure(this.state.measures.uncovered_lines)}
+ {this.renderMeasure(this.state.measures.lines_to_cover)}
+ {this.renderMeasure(this.state.measures.uncovered_conditions)}
+ {this.renderMeasure(this.state.measures.conditions_to_cover)}
+ </div>
+ </div>
+ </div>
+ </div>
+ );
+ };
+
+ renderDuplications = () => {
+ const { duplicated_lines_density: duplications } = this.state.measures;
+ if (!duplications) {
+ return null;
+ }
+ return (
+ <div className="source-viewer-measures-section">
+ <div className="source-viewer-measures-card">
+ <div className="measures">
+ <div className="measures-chart">
+ <DuplicationsRating
+ muted={duplications.value === undefined}
+ size="big"
+ value={Number(duplications.value || 0)}
+ />
+ </div>
+ <div className="measure measure-big" data-metric={duplications.metric.key}>
+ <span className="measure-value">
+ <Measure
+ metricKey={duplications.metric.key}
+ metricType={duplications.metric.type}
+ value={duplications.value}
+ />
+ </span>
+ <span className="measure-name">
+ {getLocalizedMetricName(duplications.metric, true)}
+ </span>
+ </div>
+ </div>
+
+ <div className="measures">
+ <div className="measures-list">
+ {this.renderMeasure(this.state.measures.duplicated_blocks)}
+ {this.renderMeasure(this.state.measures.duplicated_lines)}
+ </div>
+ </div>
+ </div>
+ </div>
+ );
+ };
+
+ renderTests = () => {
+ const { measures } = this.state;
+ return (
+ <div className="source-viewer-measures">
+ <div className="source-viewer-measures-section">
+ <div className="source-viewer-measures-card">
+ <div className="measures">
+ <div className="measures-list">
+ {this.renderMeasure(measures.tests)}
+ {this.renderMeasure(measures.test_success_density)}
+ {this.renderMeasure(measures.test_failures)}
+ {this.renderMeasure(measures.test_errors)}
+ {this.renderMeasure(measures.skipped_tests)}
+ {this.renderMeasure(measures.test_execution_time)}
+ </div>
+ </div>
+ </div>
+ </div>
+ </div>
+ );
+ };
+
+ renderDomain = (domain: string, measures: MeasureEnhanced[]) => {
+ return (
+ <div className="source-viewer-measures-card" key={domain}>
+ <div className="measures">
+ <div className="measures-list">
+ <div className="measure measure-one-line measure-big">
+ <span className="measure-name">{domain}</span>
+ </div>
+ {sortBy(measures.filter(measure => measure.value !== undefined), measure =>
+ getLocalizedMetricName(measure.metric)
+ ).map(measure => this.renderMeasure(measure))}
+ </div>
+ </div>
+ </div>
+ );
+ };
+
+ renderAllMeasures = () => {
+ const domains = groupBy(Object.values(this.state.measures), measure => measure.metric.domain);
+ const domainKeys = Object.keys(domains);
+ const odd = domainKeys.filter((_, index) => index % 2 === 1);
+ const even = domainKeys.filter((_, index) => index % 2 === 0);
+ return (
+ <div className="source-viewer-measures source-viewer-measures-secondary js-all-measures">
+ <div className="source-viewer-measures-section source-viewer-measures-section-big">
+ {odd.map(domain => this.renderDomain(domain, domains[domain]))}
+ </div>
+ <div className="source-viewer-measures-section source-viewer-measures-section-big">
+ {even.map(domain => this.renderDomain(domain, domains[domain]))}
+ </div>
+ </div>
+ );
+ };
+
+ render() {
+ const { branch, sourceViewerFile } = this.props;
+ const { loading } = this.state;
+
+ return (
+ <Modal contentLabel="" large={true} onRequestClose={this.props.onClose}>
+ <div className="modal-container source-viewer-measures-modal">
+ <div className="source-viewer-header-component source-viewer-measures-component">
+ <div className="source-viewer-header-component-project">
+ <QualifierIcon className="little-spacer-right" qualifier="TRK" />
+ <Link to={getProjectUrl(sourceViewerFile.project, branch)}>
+ {sourceViewerFile.projectName}
+ </Link>
+
+ {sourceViewerFile.subProject && (
+ <>
+ <QualifierIcon className="big-spacer-left little-spacer-right" qualifier="BRC" />
+ <Link to={getProjectUrl(sourceViewerFile.subProject, branch)}>
+ {sourceViewerFile.subProjectName}
+ </Link>
+ </>
+ )}
+ </div>
+
+ <div className="source-viewer-header-component-name">
+ <QualifierIcon className="little-spacer-right" qualifier={sourceViewerFile.q} />
+ {sourceViewerFile.path}
+ </div>
+ </div>
+
+ {loading ? (
+ <i className="spinner" />
+ ) : (
+ <>
+ {sourceViewerFile.q === 'UTS' ? (
+ <>
+ {this.renderTests()}
+ <MeasuresOverlayTestCases branch={branch} componentKey={sourceViewerFile.key} />
+ </>
+ ) : (
+ <div className="source-viewer-measures">
+ {this.renderLines()}
+ {this.renderIssues()}
+ {this.renderCoverage()}
+ {this.renderDuplications()}
+ </div>
+ )}
+ </>
+ )}
+
+ <div className="spacer-top">
+ {this.state.showAllMeasures ? (
+ this.renderAllMeasures()
+ ) : (
+ <a className="js-show-all-measures" href="#" onClick={this.handleAllMeasuresClick}>
+ {translate('component_viewer.show_all_measures')}
+ </a>
+ )}
+ </div>
+ </div>
+
+ <footer className="modal-foot">
+ <button className="button-link" onClick={this.handleCloseClick} type="button">
+ {translate('close')}
+ </button>
+ </footer>
+ </Modal>
+ );
+ }
+}
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayCoveredFiles.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayCoveredFiles.tsx
new file mode 100644
index 00000000000..f89a9aa408b
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayCoveredFiles.tsx
@@ -0,0 +1,115 @@
+/*
+ * 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 { Link } from 'react-router';
+import { getCoveredFiles } from '../../../api/tests';
+import { TestCase, CoveredFile } from '../../../app/types';
+import { translate, translateWithParameters } from '../../../helpers/l10n';
+import { getProjectUrl } from '../../../helpers/urls';
+import DeferredSpinner from '../../common/DeferredSpinner';
+
+interface Props {
+ testCase: TestCase;
+}
+
+interface State {
+ coveredFiles?: CoveredFile[];
+ loading: boolean;
+}
+
+export default class MeasuresOverlayCoveredFiles extends React.PureComponent<Props, State> {
+ mounted = false;
+ state: State = { loading: true };
+
+ componentDidMount() {
+ this.mounted = true;
+ this.fetchCoveredFiles();
+ }
+
+ componentDidUpdate(prevProps: Props) {
+ if (this.props.testCase.id !== prevProps.testCase.id) {
+ this.fetchCoveredFiles();
+ }
+ }
+
+ componentWillUnmount() {
+ this.mounted = false;
+ }
+
+ fetchCoveredFiles = () => {
+ this.setState({ loading: true });
+ getCoveredFiles({ testId: this.props.testCase.id }).then(
+ coveredFiles => {
+ if (this.mounted) {
+ this.setState({ coveredFiles, loading: false });
+ }
+ },
+ () => {
+ if (this.mounted) {
+ this.setState({ loading: false });
+ }
+ }
+ );
+ };
+
+ render() {
+ const { testCase } = this.props;
+ const { loading, coveredFiles } = this.state;
+
+ return (
+ <div className="source-viewer-measures-section source-viewer-measures-section-big js-selected-test">
+ <DeferredSpinner loading={loading}>
+ <div className="source-viewer-measures-card source-viewer-measures-card-fixed-height">
+ {testCase.status !== 'ERROR' &&
+ testCase.status !== 'FAILURE' &&
+ coveredFiles !== undefined && (
+ <>
+ <div className="bubble-popup-title">
+ {translate('component_viewer.transition.covers')}
+ </div>
+ {coveredFiles.length > 0
+ ? coveredFiles.map(coveredFile => (
+ <div className="bubble-popup-section" key={coveredFile.key}>
+ <Link to={getProjectUrl(coveredFile.key)}>{coveredFile.longName}</Link>
+ <span className="note spacer-left">
+ {translateWithParameters(
+ 'component_viewer.x_lines_are_covered',
+ coveredFile.coveredLines
+ )}
+ </span>
+ </div>
+ ))
+ : translate('none')}
+ </>
+ )}
+
+ {testCase.status !== 'OK' && (
+ <>
+ <div className="bubble-popup-title">{translate('component_viewer.details')}</div>
+ {testCase.message && <pre>{testCase.message}</pre>}
+ <pre>{testCase.stacktrace}</pre>
+ </>
+ )}
+ </div>
+ </DeferredSpinner>
+ </div>
+ );
+ }
+}
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayMeasure.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayMeasure.tsx
new file mode 100644
index 00000000000..7a41a9ef001
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayMeasure.tsx
@@ -0,0 +1,57 @@
+/*
+ * 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 { Metric } from '../../../app/types';
+import Measure from '../../measure/Measure';
+import IssueTypeIcon from '../../ui/IssueTypeIcon';
+import { getLocalizedMetricName } from '../../../helpers/l10n';
+
+export interface MeasureWithMetric {
+ metric: Metric;
+ value?: string;
+}
+
+interface Props {
+ measure: MeasureWithMetric;
+}
+
+export default function MeasuresOverlayMeasure({ measure }: Props) {
+ return (
+ <div
+ className="measure measure-one-line"
+ data-metric={measure.metric.key}
+ key={measure.metric.key}>
+ <span className="measure-name">
+ {['bugs', 'vulnerabilities', 'code_smells'].includes(measure.metric.key) && (
+ <IssueTypeIcon className="little-spacer-right" query={measure.metric.key} />
+ )}
+ {getLocalizedMetricName(measure.metric)}
+ </span>
+ <span className="measure-value">
+ <Measure
+ metricKey={measure.metric.key}
+ metricType={measure.metric.type}
+ small={true}
+ value={measure.value}
+ />
+ </span>
+ </div>
+ );
+}
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayTestCase.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayTestCase.tsx
new file mode 100644
index 00000000000..17b3e455ba0
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayTestCase.tsx
@@ -0,0 +1,62 @@
+/*
+ * 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 { TestCase } from '../../../app/types';
+import TestStatusIcon from '../../shared/TestStatusIcon';
+
+interface Props {
+ onClick: (testId: string) => void;
+ testCase: TestCase;
+}
+
+export default class MeasuresOverlayTestCase extends React.PureComponent<Props> {
+ handleTestCaseClick = (event: React.SyntheticEvent<HTMLAnchorElement>) => {
+ event.preventDefault();
+ event.currentTarget.blur();
+ this.props.onClick(this.props.testCase.id);
+ };
+
+ render() {
+ const { testCase } = this.props;
+ const { status } = testCase;
+ const hasAdditionalData = status !== 'OK' || (status === 'OK' && testCase.coveredLines);
+
+ return (
+ <tr>
+ <td className="source-viewer-test-status">
+ <TestStatusIcon status={status} />
+ </td>
+ <td className="source-viewer-test-duration note">
+ {status !== 'SKIPPED' && `${testCase.durationInMs}ms`}
+ </td>
+ <td className="source-viewer-test-name">
+ {hasAdditionalData ? (
+ <a className="js-show-test" href="#" onClick={this.handleTestCaseClick}>
+ {testCase.name}
+ </a>
+ ) : (
+ testCase.name
+ )}
+ </td>
+ <td className="source-viewer-test-covered-lines note">{testCase.coveredLines}</td>
+ </tr>
+ );
+ }
+}
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayTestCases.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayTestCases.tsx
new file mode 100644
index 00000000000..64dca78addc
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/MeasuresOverlayTestCases.tsx
@@ -0,0 +1,181 @@
+/*
+ * 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 { orderBy } from 'lodash';
+import MeasuresOverlayCoveredFiles from './MeasuresOverlayCoveredFiles';
+import MeasuresOverlayTestCase from './MeasuresOverlayTestCase';
+import { getTests } from '../../../api/tests';
+import { TestCase } from '../../../app/types';
+import { translate } from '../../../helpers/l10n';
+
+interface Props {
+ branch: string | undefined;
+ componentKey: string;
+}
+
+interface State {
+ loading: boolean;
+ selectedTestId?: string;
+ sort?: string;
+ sortAsc?: boolean;
+ testCases?: TestCase[];
+}
+
+export default class MeasuresOverlayTestCases extends React.PureComponent<Props, State> {
+ mounted = false;
+ state: State = { loading: true };
+
+ componentDidMount() {
+ this.mounted = true;
+ this.fetchTests();
+ }
+
+ componentDidUpdate(prevProps: Props) {
+ if (
+ prevProps.branch !== this.props.branch ||
+ prevProps.componentKey !== this.props.componentKey
+ ) {
+ this.fetchTests();
+ }
+ }
+
+ componentWillUnmount() {
+ this.mounted = false;
+ }
+
+ fetchTests = () => {
+ // TODO implement pagination one day...
+ this.setState({ loading: true });
+ getTests({ branch: this.props.branch, ps: 500, testFileKey: this.props.componentKey }).then(
+ ({ tests: testCases }) => {
+ if (this.mounted) {
+ this.setState({ loading: false, testCases });
+ }
+ },
+ () => {
+ if (this.mounted) {
+ this.setState({ loading: false });
+ }
+ }
+ );
+ };
+
+ handleTestCasesSortClick = (event: React.SyntheticEvent<HTMLAnchorElement>) => {
+ event.preventDefault();
+ event.currentTarget.blur();
+ const { sort } = event.currentTarget.dataset;
+ if (sort) {
+ this.setState((state: State) => ({
+ sort,
+ sortAsc: sort === state.sort ? !state.sortAsc : true
+ }));
+ }
+ };
+
+ handleTestCaseClick = (selectedTestId: string) => {
+ this.setState({ selectedTestId });
+ };
+
+ render() {
+ const { selectedTestId, sort = 'name', sortAsc = true, testCases } = this.state;
+
+ if (!testCases) {
+ return null;
+ }
+
+ const selectedTest = testCases.find(test => test.id === selectedTestId);
+
+ return (
+ <div className="source-viewer-measures">
+ <div className="source-viewer-measures-section source-viewer-measures-section-big">
+ <div className="source-viewer-measures-card source-viewer-measures-card-fixed-height js-test-list">
+ <div className="measures">
+ <table className="source-viewer-tests-list">
+ <tbody>
+ <tr>
+ <td className="source-viewer-test-status note" colSpan={3}>
+ {translate('component_viewer.measure_section.unit_tests')}
+ <br />
+ <span className="spacer-right">
+ {translate('component_viewer.tests.ordered_by')}
+ </span>
+ <a
+ className={classNames('js-sort-tests-by-duration', {
+ 'active-link': sort === 'duration'
+ })}
+ data-sort="duration"
+ href="#"
+ onClick={this.handleTestCasesSortClick}>
+ {translate('component_viewer.tests.duration')}
+ </a>
+ <span className="slash-separator" />
+ <a
+ className={classNames('js-sort-tests-by-name', {
+ 'active-link': sort === 'name'
+ })}
+ data-sort="name"
+ href="#"
+ onClick={this.handleTestCasesSortClick}>
+ {translate('component_viewer.tests.test_name')}
+ </a>
+ <span className="slash-separator" />
+ <a
+ className={classNames('js-sort-tests-by-status', {
+ 'active-link': sort === 'status'
+ })}
+ data-sort="status"
+ href="#"
+ onClick={this.handleTestCasesSortClick}>
+ {translate('component_viewer.tests.status')}
+ </a>
+ </td>
+ <td className="source-viewer-test-covered-lines note">
+ {translate('component_viewer.covered_lines')}
+ </td>
+ </tr>
+ {sortTestCases(testCases, sort, sortAsc).map(testCase => (
+ <MeasuresOverlayTestCase
+ key={testCase.id}
+ onClick={this.handleTestCaseClick}
+ testCase={testCase}
+ />
+ ))}
+ </tbody>
+ </table>
+ </div>
+ </div>
+ </div>
+ {selectedTest && <MeasuresOverlayCoveredFiles testCase={selectedTest} />}
+ </div>
+ );
+ }
+}
+
+function sortTestCases(testCases: TestCase[], sort: string, sortAsc: boolean) {
+ const mainOrder = sortAsc ? 'asc' : 'desc';
+ if (sort === 'duration') {
+ return orderBy(testCases, ['durationInMs', 'name'], [mainOrder, 'asc']);
+ } else if (sort === 'status') {
+ return orderBy(testCases, ['status', 'name'], [mainOrder, 'asc']);
+ } else {
+ return orderBy(testCases, ['name'], [mainOrder]);
+ }
+}
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlay-test.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlay-test.tsx
new file mode 100644
index 00000000000..1f94f1375d3
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlay-test.tsx
@@ -0,0 +1,172 @@
+/*
+ * 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 MeasuresOverlay from '../MeasuresOverlay';
+import { SourceViewerFile } from '../../../../app/types';
+import { waitAndUpdate, click } from '../../../../helpers/testUtils';
+
+jest.mock('../../../../api/issues', () => ({
+ getFacets: () =>
+ Promise.resolve({
+ facets: [
+ {
+ property: 'types',
+ values: [
+ { val: 'CODE_SMELL', count: 2 },
+ { val: 'BUG', count: 1 },
+ { val: 'VULNERABILITY', count: 0 }
+ ]
+ },
+ {
+ property: 'severities',
+ values: [
+ { val: 'MAJOR', count: 1 },
+ { val: 'INFO', count: 2 },
+ { val: 'MINOR', count: 3 },
+ { val: 'CRITICAL', count: 4 },
+ { val: 'BLOCKER', count: 5 }
+ ]
+ },
+ {
+ property: 'tags',
+ values: [
+ { val: 'bad-practice', count: 1 },
+ { val: 'cert', count: 3 },
+ { val: 'design', count: 1 }
+ ]
+ }
+ ]
+ })
+}));
+
+jest.mock('../../../../api/measures', () => ({
+ getMeasures: () =>
+ Promise.resolve([
+ { metric: 'vulnerabilities', value: '0' },
+ { metric: 'complexity', value: '1' },
+ { metric: 'test_errors', value: '1' },
+ { metric: 'comment_lines_density', value: '20.0' },
+ { metric: 'wont_fix_issues', value: '0' },
+ { metric: 'uncovered_lines', value: '1' },
+ { metric: 'functions', value: '1' },
+ { metric: 'duplicated_files', value: '1' },
+ { metric: 'duplicated_blocks', value: '3' },
+ { metric: 'line_coverage', value: '75.0' },
+ { metric: 'duplicated_lines_density', value: '0.0' },
+ { metric: 'comment_lines', value: '2' },
+ { metric: 'ncloc', value: '8' },
+ { metric: 'reliability_rating', value: '1.0' },
+ { metric: 'false_positive_issues', value: '0' },
+ { metric: 'reliability_remediation_effort', value: '0' },
+ { metric: 'code_smells', value: '2' },
+ { metric: 'security_rating', value: '1.0' },
+ { metric: 'test_success_density', value: '100.0' },
+ { metric: 'cognitive_complexity', value: '0' },
+ { metric: 'files', value: '1' },
+ { metric: 'duplicated_lines', value: '0' },
+ { metric: 'lines', value: '18' },
+ { metric: 'classes', value: '1' },
+ { metric: 'bugs', value: '0' },
+ { metric: 'lines_to_cover', value: '4' },
+ { metric: 'sqale_index', value: '40' },
+ { metric: 'sqale_debt_ratio', value: '16.7' },
+ { metric: 'coverage', value: '75.0' },
+ { metric: 'security_remediation_effort', value: '0' },
+ { metric: 'statements', value: '3' },
+ { metric: 'skipped_tests', value: '0' },
+ { metric: 'test_failures', value: '0' }
+ ])
+}));
+
+jest.mock('../../../../api/metrics', () => ({
+ getAllMetrics: () =>
+ Promise.resolve([
+ { key: 'vulnerabilities', type: 'INT', domain: 'Security' },
+ { key: 'complexity', type: 'INT', domain: 'Complexity' },
+ { key: 'test_errors', type: 'INT', domain: 'Tests' },
+ { key: 'comment_lines_density', type: 'PERCENT', domain: 'Size' },
+ { key: 'wont_fix_issues', type: 'INT', domain: 'Issues' },
+ { key: 'uncovered_lines', type: 'INT', domain: 'Coverage' },
+ { key: 'functions', type: 'INT', domain: 'Size' },
+ { key: 'duplicated_files', type: 'INT', domain: 'Duplications' },
+ { key: 'duplicated_blocks', type: 'INT', domain: 'Duplications' },
+ { key: 'line_coverage', type: 'PERCENT', domain: 'Coverage' },
+ { key: 'duplicated_lines_density', type: 'PERCENT', domain: 'Duplications' },
+ { key: 'comment_lines', type: 'INT', domain: 'Size' },
+ { key: 'ncloc', type: 'INT', domain: 'Size' },
+ { key: 'reliability_rating', type: 'RATING', domain: 'Reliability' },
+ { key: 'false_positive_issues', type: 'INT', domain: 'Issues' },
+ { key: 'code_smells', type: 'INT', domain: 'Maintainability' },
+ { key: 'security_rating', type: 'RATING', domain: 'Security' },
+ { key: 'test_success_density', type: 'PERCENT', domain: 'Tests' },
+ { key: 'cognitive_complexity', type: 'INT', domain: 'Complexity' },
+ { key: 'files', type: 'INT', domain: 'Size' },
+ { key: 'duplicated_lines', type: 'INT', domain: 'Duplications' },
+ { key: 'lines', type: 'INT', domain: 'Size' },
+ { key: 'classes', type: 'INT', domain: 'Size' },
+ { key: 'bugs', type: 'INT', domain: 'Reliability' },
+ { key: 'lines_to_cover', type: 'INT', domain: 'Coverage' },
+ { key: 'sqale_index', type: 'WORK_DUR', domain: 'Maintainability' },
+ { key: 'sqale_debt_ratio', type: 'PERCENT', domain: 'Maintainability' },
+ { key: 'coverage', type: 'PERCENT', domain: 'Coverage' },
+ { key: 'statements', type: 'INT', domain: 'Size' },
+ { key: 'skipped_tests', type: 'INT', domain: 'Tests' },
+ { key: 'test_failures', type: 'INT', domain: 'Tests' },
+ // next two must be filtered out
+ { key: 'data', type: 'DATA' },
+ { key: 'hidden', hidden: true }
+ ])
+}));
+
+const sourceViewerFile: SourceViewerFile = {
+ key: 'component-key',
+ measures: {},
+ path: 'src/file.js',
+ project: 'project-key',
+ projectName: 'Project Name',
+ q: 'FIL',
+ subProject: 'sub-project-key',
+ subProjectName: 'Sub-Project Name',
+ uuid: 'abcd123'
+};
+
+it('should render source file', async () => {
+ const wrapper = shallow(
+ <MeasuresOverlay branch="branch" onClose={jest.fn()} sourceViewerFile={sourceViewerFile} />
+ );
+ await waitAndUpdate(wrapper);
+ expect(wrapper).toMatchSnapshot();
+
+ click(wrapper.find('.js-show-all-measures'));
+ expect(wrapper).toMatchSnapshot();
+});
+
+it('should render test file', async () => {
+ const wrapper = shallow(
+ <MeasuresOverlay
+ branch="branch"
+ onClose={jest.fn()}
+ sourceViewerFile={{ ...sourceViewerFile, q: 'UTS' }}
+ />
+ );
+ await waitAndUpdate(wrapper);
+ expect(wrapper).toMatchSnapshot();
+});
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayCoveredFiles-test.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayCoveredFiles-test.tsx
new file mode 100644
index 00000000000..4be9b29f714
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayCoveredFiles-test.tsx
@@ -0,0 +1,55 @@
+/*
+ * 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 MeasuresOverlayCoveredFiles from '../MeasuresOverlayCoveredFiles';
+import { waitAndUpdate } from '../../../../helpers/testUtils';
+
+jest.mock('../../../../api/tests', () => ({
+ getCoveredFiles: () =>
+ Promise.resolve([{ key: 'project:src/file.js', longName: 'src/file.js', coveredLines: 3 }])
+}));
+
+const testCase = {
+ coveredLines: 3,
+ durationInMs: 1,
+ fileId: 'abcd',
+ fileKey: 'project:test.js',
+ fileName: 'test.js',
+ id: 'test-abcd',
+ name: 'should work',
+ status: 'OK'
+};
+
+it('should render OK test', async () => {
+ const wrapper = shallow(<MeasuresOverlayCoveredFiles testCase={testCase} />);
+ await waitAndUpdate(wrapper);
+ expect(wrapper).toMatchSnapshot();
+});
+
+it('should render ERROR test', async () => {
+ const wrapper = shallow(
+ <MeasuresOverlayCoveredFiles
+ testCase={{ ...testCase, status: 'ERROR', message: 'Something failed' }}
+ />
+ );
+ await waitAndUpdate(wrapper);
+ expect(wrapper).toMatchSnapshot();
+});
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayMeasure-test.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayMeasure-test.tsx
new file mode 100644
index 00000000000..c16c841c841
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayMeasure-test.tsx
@@ -0,0 +1,48 @@
+/*
+ * 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 MeasuresOverlayMeasure from '../MeasuresOverlayMeasure';
+
+it('should render', () => {
+ expect(
+ shallow(
+ <MeasuresOverlayMeasure
+ measure={{
+ metric: { id: '1', key: 'coverage', name: 'Coverage', type: 'PERCENT' },
+ value: '72'
+ }}
+ />
+ )
+ ).toMatchSnapshot();
+});
+
+it('should render issues icon', () => {
+ expect(
+ shallow(
+ <MeasuresOverlayMeasure
+ measure={{
+ metric: { id: '1', key: 'bugs', name: 'Bugs', type: 'INT' },
+ value: '2'
+ }}
+ />
+ )
+ ).toMatchSnapshot();
+});
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayTestCase-test.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayTestCase-test.tsx
new file mode 100644
index 00000000000..6c51c7c3433
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayTestCase-test.tsx
@@ -0,0 +1,42 @@
+/*
+ * 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 MeasuresOverlayTestCase from '../MeasuresOverlayTestCase';
+import { click } from '../../../../helpers/testUtils';
+
+const testCase = {
+ coveredLines: 3,
+ durationInMs: 1,
+ fileId: 'abcd',
+ fileKey: 'project:test.js',
+ fileName: 'test.js',
+ id: 'test-abcd',
+ name: 'should work',
+ status: 'OK'
+};
+
+it('should render', () => {
+ const onClick = jest.fn();
+ const wrapper = shallow(<MeasuresOverlayTestCase onClick={onClick} testCase={testCase} />);
+ expect(wrapper).toMatchSnapshot();
+ click(wrapper.find('a'));
+ expect(onClick).toBeCalledWith('test-abcd');
+});
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayTestCases-test.tsx b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayTestCases-test.tsx
new file mode 100644
index 00000000000..72226708c6f
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/MeasuresOverlayTestCases-test.tsx
@@ -0,0 +1,74 @@
+/*
+ * 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 MeasuresOverlayTestCases from '../MeasuresOverlayTestCases';
+import { waitAndUpdate, click } from '../../../../helpers/testUtils';
+
+jest.mock('../../../../api/tests', () => ({
+ getTests: () =>
+ Promise.resolve({
+ tests: [
+ {
+ id: 'AWGub2mGGZxsAttCZwQy',
+ name: 'testAdd_WhichFails',
+ fileKey: 'test:fake-project-for-tests:src/test/java/bar/SimplestTest.java',
+ fileName: 'src/test/java/bar/SimplestTest.java',
+ status: 'FAILURE',
+ durationInMs: 6,
+ coveredLines: 3,
+ message: 'expected:<9> but was:<2>',
+ stacktrace:
+ 'java.lang.AssertionError: expected:<9> but was:<2>\n\tat org.junit.Assert.fail(Assert.java:93)\n\tat org.junit.Assert.failNotEquals(Assert.java:647)'
+ },
+ {
+ id: 'AWGub2mGGZxsAttCZwQz',
+ name: 'testAdd_InError',
+ fileKey: 'test:fake-project-for-tests:src/test/java/bar/SimplestTest.java',
+ fileName: 'src/test/java/bar/SimplestTest.java',
+ status: 'ERROR',
+ durationInMs: 2,
+ coveredLines: 3
+ },
+ {
+ id: 'AWGub2mFGZxsAttCZwQx',
+ name: 'testAdd',
+ fileKey: 'test:fake-project-for-tests:src/test/java/bar/SimplestTest.java',
+ fileName: 'src/test/java/bar/SimplestTest.java',
+ status: 'OK',
+ durationInMs: 8,
+ coveredLines: 3
+ }
+ ]
+ })
+}));
+
+it('should render', async () => {
+ const wrapper = shallow(
+ <MeasuresOverlayTestCases branch="branch" componentKey="component-key" />
+ );
+ await waitAndUpdate(wrapper);
+ expect(wrapper).toMatchSnapshot();
+
+ click(wrapper.find('.js-sort-tests-by-duration'), {
+ currentTarget: { blur() {}, dataset: { sort: 'duration' } }
+ });
+ expect(wrapper).toMatchSnapshot();
+});
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlay-test.tsx.snap b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlay-test.tsx.snap
new file mode 100644
index 00000000000..bd7eb4c902b
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlay-test.tsx.snap
@@ -0,0 +1,1535 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`should render source file 1`] = `
+<Modal
+ contentLabel=""
+ large={true}
+ onRequestClose={[MockFunction]}
+>
+ <div
+ className="modal-container source-viewer-measures-modal"
+ >
+ <div
+ className="source-viewer-header-component source-viewer-measures-component"
+ >
+ <div
+ className="source-viewer-header-component-project"
+ >
+ <QualifierIcon
+ className="little-spacer-right"
+ qualifier="TRK"
+ />
+ <Link
+ onlyActiveOnIndex={false}
+ style={Object {}}
+ to={
+ Object {
+ "pathname": "/dashboard",
+ "query": Object {
+ "branch": "branch",
+ "id": "project-key",
+ },
+ }
+ }
+ >
+ Project Name
+ </Link>
+ <React.Fragment>
+ <QualifierIcon
+ className="big-spacer-left little-spacer-right"
+ qualifier="BRC"
+ />
+ <Link
+ onlyActiveOnIndex={false}
+ style={Object {}}
+ to={
+ Object {
+ "pathname": "/dashboard",
+ "query": Object {
+ "branch": "branch",
+ "id": "sub-project-key",
+ },
+ }
+ }
+ >
+ Sub-Project Name
+ </Link>
+ </React.Fragment>
+ </div>
+ <div
+ className="source-viewer-header-component-name"
+ >
+ <QualifierIcon
+ className="little-spacer-right"
+ qualifier="FIL"
+ />
+ src/file.js
+ </div>
+ </div>
+ <React.Fragment>
+ <div
+ className="source-viewer-measures"
+ >
+ <div
+ className="source-viewer-measures-section"
+ >
+ <div
+ className="source-viewer-measures-card"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <MeasuresOverlayMeasure
+ key="lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "lines",
+ "type": "INT",
+ },
+ "value": "18",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="ncloc"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "ncloc",
+ "type": "INT",
+ },
+ "value": "8",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="comment_lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "comment_lines",
+ "type": "INT",
+ },
+ "value": "2",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="comment_lines_density"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "comment_lines_density",
+ "type": "PERCENT",
+ },
+ "value": "20.0",
+ }
+ }
+ />
+ </div>
+ </div>
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <MeasuresOverlayMeasure
+ key="cognitive_complexity"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Complexity",
+ "key": "cognitive_complexity",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="complexity"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Complexity",
+ "key": "complexity",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-section"
+ >
+ <div
+ className="source-viewer-measures-card"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measure measure-big"
+ data-metric="sqale_index"
+ >
+ <span
+ className="measure-value"
+ >
+ <Measure
+ metricKey="sqale_index"
+ metricType="WORK_DUR"
+ value="40"
+ />
+ </span>
+ <span
+ className="measure-name"
+ >
+ sqale_index
+ </span>
+ </div>
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-section"
+ >
+ <div
+ className="source-viewer-measures-card"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-chart"
+ >
+ <CoverageRating
+ size="big"
+ value="75.0"
+ />
+ </div>
+ <div
+ className="measure measure-big"
+ data-metric="coverage"
+ >
+ <span
+ className="measure-value"
+ >
+ <Measure
+ metricKey="coverage"
+ metricType="PERCENT"
+ value="75.0"
+ />
+ </span>
+ <span
+ className="measure-name"
+ >
+ coverage
+ </span>
+ </div>
+ </div>
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <MeasuresOverlayMeasure
+ key="uncovered_lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Coverage",
+ "key": "uncovered_lines",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="lines_to_cover"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Coverage",
+ "key": "lines_to_cover",
+ "type": "INT",
+ },
+ "value": "4",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-section"
+ >
+ <div
+ className="source-viewer-measures-card"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-chart"
+ >
+ <DuplicationsRating
+ muted={false}
+ size="big"
+ value={0}
+ />
+ </div>
+ <div
+ className="measure measure-big"
+ data-metric="duplicated_lines_density"
+ >
+ <span
+ className="measure-value"
+ >
+ <Measure
+ metricKey="duplicated_lines_density"
+ metricType="PERCENT"
+ value="0.0"
+ />
+ </span>
+ <span
+ className="measure-name"
+ >
+ duplicated_lines_density
+ </span>
+ </div>
+ </div>
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <MeasuresOverlayMeasure
+ key="duplicated_blocks"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Duplications",
+ "key": "duplicated_blocks",
+ "type": "INT",
+ },
+ "value": "3",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="duplicated_lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Duplications",
+ "key": "duplicated_lines",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ </div>
+ </div>
+ </React.Fragment>
+ <div
+ className="spacer-top"
+ >
+ <a
+ className="js-show-all-measures"
+ href="#"
+ onClick={[Function]}
+ >
+ component_viewer.show_all_measures
+ </a>
+ </div>
+ </div>
+ <footer
+ className="modal-foot"
+ >
+ <button
+ className="button-link"
+ onClick={[Function]}
+ type="button"
+ >
+ close
+ </button>
+ </footer>
+</Modal>
+`;
+
+exports[`should render source file 2`] = `
+<Modal
+ contentLabel=""
+ large={true}
+ onRequestClose={[MockFunction]}
+>
+ <div
+ className="modal-container source-viewer-measures-modal"
+ >
+ <div
+ className="source-viewer-header-component source-viewer-measures-component"
+ >
+ <div
+ className="source-viewer-header-component-project"
+ >
+ <QualifierIcon
+ className="little-spacer-right"
+ qualifier="TRK"
+ />
+ <Link
+ onlyActiveOnIndex={false}
+ style={Object {}}
+ to={
+ Object {
+ "pathname": "/dashboard",
+ "query": Object {
+ "branch": "branch",
+ "id": "project-key",
+ },
+ }
+ }
+ >
+ Project Name
+ </Link>
+ <React.Fragment>
+ <QualifierIcon
+ className="big-spacer-left little-spacer-right"
+ qualifier="BRC"
+ />
+ <Link
+ onlyActiveOnIndex={false}
+ style={Object {}}
+ to={
+ Object {
+ "pathname": "/dashboard",
+ "query": Object {
+ "branch": "branch",
+ "id": "sub-project-key",
+ },
+ }
+ }
+ >
+ Sub-Project Name
+ </Link>
+ </React.Fragment>
+ </div>
+ <div
+ className="source-viewer-header-component-name"
+ >
+ <QualifierIcon
+ className="little-spacer-right"
+ qualifier="FIL"
+ />
+ src/file.js
+ </div>
+ </div>
+ <React.Fragment>
+ <div
+ className="source-viewer-measures"
+ >
+ <div
+ className="source-viewer-measures-section"
+ >
+ <div
+ className="source-viewer-measures-card"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <MeasuresOverlayMeasure
+ key="lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "lines",
+ "type": "INT",
+ },
+ "value": "18",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="ncloc"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "ncloc",
+ "type": "INT",
+ },
+ "value": "8",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="comment_lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "comment_lines",
+ "type": "INT",
+ },
+ "value": "2",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="comment_lines_density"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "comment_lines_density",
+ "type": "PERCENT",
+ },
+ "value": "20.0",
+ }
+ }
+ />
+ </div>
+ </div>
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <MeasuresOverlayMeasure
+ key="cognitive_complexity"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Complexity",
+ "key": "cognitive_complexity",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="complexity"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Complexity",
+ "key": "complexity",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-section"
+ >
+ <div
+ className="source-viewer-measures-card"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measure measure-big"
+ data-metric="sqale_index"
+ >
+ <span
+ className="measure-value"
+ >
+ <Measure
+ metricKey="sqale_index"
+ metricType="WORK_DUR"
+ value="40"
+ />
+ </span>
+ <span
+ className="measure-name"
+ >
+ sqale_index
+ </span>
+ </div>
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-section"
+ >
+ <div
+ className="source-viewer-measures-card"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-chart"
+ >
+ <CoverageRating
+ size="big"
+ value="75.0"
+ />
+ </div>
+ <div
+ className="measure measure-big"
+ data-metric="coverage"
+ >
+ <span
+ className="measure-value"
+ >
+ <Measure
+ metricKey="coverage"
+ metricType="PERCENT"
+ value="75.0"
+ />
+ </span>
+ <span
+ className="measure-name"
+ >
+ coverage
+ </span>
+ </div>
+ </div>
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <MeasuresOverlayMeasure
+ key="uncovered_lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Coverage",
+ "key": "uncovered_lines",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="lines_to_cover"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Coverage",
+ "key": "lines_to_cover",
+ "type": "INT",
+ },
+ "value": "4",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-section"
+ >
+ <div
+ className="source-viewer-measures-card"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-chart"
+ >
+ <DuplicationsRating
+ muted={false}
+ size="big"
+ value={0}
+ />
+ </div>
+ <div
+ className="measure measure-big"
+ data-metric="duplicated_lines_density"
+ >
+ <span
+ className="measure-value"
+ >
+ <Measure
+ metricKey="duplicated_lines_density"
+ metricType="PERCENT"
+ value="0.0"
+ />
+ </span>
+ <span
+ className="measure-name"
+ >
+ duplicated_lines_density
+ </span>
+ </div>
+ </div>
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <MeasuresOverlayMeasure
+ key="duplicated_blocks"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Duplications",
+ "key": "duplicated_blocks",
+ "type": "INT",
+ },
+ "value": "3",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="duplicated_lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Duplications",
+ "key": "duplicated_lines",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ </div>
+ </div>
+ </React.Fragment>
+ <div
+ className="spacer-top"
+ >
+ <div
+ className="source-viewer-measures source-viewer-measures-secondary js-all-measures"
+ >
+ <div
+ className="source-viewer-measures-section source-viewer-measures-section-big"
+ >
+ <div
+ className="source-viewer-measures-card"
+ key="Complexity"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <div
+ className="measure measure-one-line measure-big"
+ >
+ <span
+ className="measure-name"
+ >
+ Complexity
+ </span>
+ </div>
+ <MeasuresOverlayMeasure
+ key="cognitive_complexity"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Complexity",
+ "key": "cognitive_complexity",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="complexity"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Complexity",
+ "key": "complexity",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-card"
+ key="Size"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <div
+ className="measure measure-one-line measure-big"
+ >
+ <span
+ className="measure-name"
+ >
+ Size
+ </span>
+ </div>
+ <MeasuresOverlayMeasure
+ key="classes"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "classes",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="comment_lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "comment_lines",
+ "type": "INT",
+ },
+ "value": "2",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="comment_lines_density"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "comment_lines_density",
+ "type": "PERCENT",
+ },
+ "value": "20.0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="files"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "files",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="functions"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "functions",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "lines",
+ "type": "INT",
+ },
+ "value": "18",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="ncloc"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "ncloc",
+ "type": "INT",
+ },
+ "value": "8",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="statements"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Size",
+ "key": "statements",
+ "type": "INT",
+ },
+ "value": "3",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-card"
+ key="Coverage"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <div
+ className="measure measure-one-line measure-big"
+ >
+ <span
+ className="measure-name"
+ >
+ Coverage
+ </span>
+ </div>
+ <MeasuresOverlayMeasure
+ key="coverage"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Coverage",
+ "key": "coverage",
+ "type": "PERCENT",
+ },
+ "value": "75.0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="line_coverage"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Coverage",
+ "key": "line_coverage",
+ "type": "PERCENT",
+ },
+ "value": "75.0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="lines_to_cover"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Coverage",
+ "key": "lines_to_cover",
+ "type": "INT",
+ },
+ "value": "4",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="uncovered_lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Coverage",
+ "key": "uncovered_lines",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-card"
+ key="Reliability"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <div
+ className="measure measure-one-line measure-big"
+ >
+ <span
+ className="measure-name"
+ >
+ Reliability
+ </span>
+ </div>
+ <MeasuresOverlayMeasure
+ key="bugs"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Reliability",
+ "key": "bugs",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="reliability_rating"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Reliability",
+ "key": "reliability_rating",
+ "type": "RATING",
+ },
+ "value": "1.0",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-section source-viewer-measures-section-big"
+ >
+ <div
+ className="source-viewer-measures-card"
+ key="Security"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <div
+ className="measure measure-one-line measure-big"
+ >
+ <span
+ className="measure-name"
+ >
+ Security
+ </span>
+ </div>
+ <MeasuresOverlayMeasure
+ key="security_rating"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Security",
+ "key": "security_rating",
+ "type": "RATING",
+ },
+ "value": "1.0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="vulnerabilities"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Security",
+ "key": "vulnerabilities",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-card"
+ key="Tests"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <div
+ className="measure measure-one-line measure-big"
+ >
+ <span
+ className="measure-name"
+ >
+ Tests
+ </span>
+ </div>
+ <MeasuresOverlayMeasure
+ key="skipped_tests"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Tests",
+ "key": "skipped_tests",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="test_errors"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Tests",
+ "key": "test_errors",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="test_failures"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Tests",
+ "key": "test_failures",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="test_success_density"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Tests",
+ "key": "test_success_density",
+ "type": "PERCENT",
+ },
+ "value": "100.0",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-card"
+ key="Issues"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <div
+ className="measure measure-one-line measure-big"
+ >
+ <span
+ className="measure-name"
+ >
+ Issues
+ </span>
+ </div>
+ <MeasuresOverlayMeasure
+ key="false_positive_issues"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Issues",
+ "key": "false_positive_issues",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="wont_fix_issues"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Issues",
+ "key": "wont_fix_issues",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-card"
+ key="Duplications"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <div
+ className="measure measure-one-line measure-big"
+ >
+ <span
+ className="measure-name"
+ >
+ Duplications
+ </span>
+ </div>
+ <MeasuresOverlayMeasure
+ key="duplicated_blocks"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Duplications",
+ "key": "duplicated_blocks",
+ "type": "INT",
+ },
+ "value": "3",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="duplicated_files"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Duplications",
+ "key": "duplicated_files",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="duplicated_lines"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Duplications",
+ "key": "duplicated_lines",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="duplicated_lines_density"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Duplications",
+ "key": "duplicated_lines_density",
+ "type": "PERCENT",
+ },
+ "value": "0.0",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ <div
+ className="source-viewer-measures-card"
+ key="Maintainability"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <div
+ className="measure measure-one-line measure-big"
+ >
+ <span
+ className="measure-name"
+ >
+ Maintainability
+ </span>
+ </div>
+ <MeasuresOverlayMeasure
+ key="code_smells"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Maintainability",
+ "key": "code_smells",
+ "type": "INT",
+ },
+ "value": "2",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="sqale_debt_ratio"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Maintainability",
+ "key": "sqale_debt_ratio",
+ "type": "PERCENT",
+ },
+ "value": "16.7",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="sqale_index"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Maintainability",
+ "key": "sqale_index",
+ "type": "WORK_DUR",
+ },
+ "value": "40",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ </div>
+ </div>
+ </div>
+ </div>
+ <footer
+ className="modal-foot"
+ >
+ <button
+ className="button-link"
+ onClick={[Function]}
+ type="button"
+ >
+ close
+ </button>
+ </footer>
+</Modal>
+`;
+
+exports[`should render test file 1`] = `
+<Modal
+ contentLabel=""
+ large={true}
+ onRequestClose={[MockFunction]}
+>
+ <div
+ className="modal-container source-viewer-measures-modal"
+ >
+ <div
+ className="source-viewer-header-component source-viewer-measures-component"
+ >
+ <div
+ className="source-viewer-header-component-project"
+ >
+ <QualifierIcon
+ className="little-spacer-right"
+ qualifier="TRK"
+ />
+ <Link
+ onlyActiveOnIndex={false}
+ style={Object {}}
+ to={
+ Object {
+ "pathname": "/dashboard",
+ "query": Object {
+ "branch": "branch",
+ "id": "project-key",
+ },
+ }
+ }
+ >
+ Project Name
+ </Link>
+ <React.Fragment>
+ <QualifierIcon
+ className="big-spacer-left little-spacer-right"
+ qualifier="BRC"
+ />
+ <Link
+ onlyActiveOnIndex={false}
+ style={Object {}}
+ to={
+ Object {
+ "pathname": "/dashboard",
+ "query": Object {
+ "branch": "branch",
+ "id": "sub-project-key",
+ },
+ }
+ }
+ >
+ Sub-Project Name
+ </Link>
+ </React.Fragment>
+ </div>
+ <div
+ className="source-viewer-header-component-name"
+ >
+ <QualifierIcon
+ className="little-spacer-right"
+ qualifier="UTS"
+ />
+ src/file.js
+ </div>
+ </div>
+ <React.Fragment>
+ <React.Fragment>
+ <div
+ className="source-viewer-measures"
+ >
+ <div
+ className="source-viewer-measures-section"
+ >
+ <div
+ className="source-viewer-measures-card"
+ >
+ <div
+ className="measures"
+ >
+ <div
+ className="measures-list"
+ >
+ <MeasuresOverlayMeasure
+ key="test_success_density"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Tests",
+ "key": "test_success_density",
+ "type": "PERCENT",
+ },
+ "value": "100.0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="test_failures"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Tests",
+ "key": "test_failures",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="test_errors"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Tests",
+ "key": "test_errors",
+ "type": "INT",
+ },
+ "value": "1",
+ }
+ }
+ />
+ <MeasuresOverlayMeasure
+ key="skipped_tests"
+ measure={
+ Object {
+ "metric": Object {
+ "domain": "Tests",
+ "key": "skipped_tests",
+ "type": "INT",
+ },
+ "value": "0",
+ }
+ }
+ />
+ </div>
+ </div>
+ </div>
+ </div>
+ </div>
+ <MeasuresOverlayTestCases
+ branch="branch"
+ componentKey="component-key"
+ />
+ </React.Fragment>
+ </React.Fragment>
+ <div
+ className="spacer-top"
+ >
+ <a
+ className="js-show-all-measures"
+ href="#"
+ onClick={[Function]}
+ >
+ component_viewer.show_all_measures
+ </a>
+ </div>
+ </div>
+ <footer
+ className="modal-foot"
+ >
+ <button
+ className="button-link"
+ onClick={[Function]}
+ type="button"
+ >
+ close
+ </button>
+ </footer>
+</Modal>
+`;
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayCoveredFiles-test.tsx.snap b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayCoveredFiles-test.tsx.snap
new file mode 100644
index 00000000000..c578435451a
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayCoveredFiles-test.tsx.snap
@@ -0,0 +1,76 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`should render ERROR test 1`] = `
+<div
+ className="source-viewer-measures-section source-viewer-measures-section-big js-selected-test"
+>
+ <DeferredSpinner
+ loading={false}
+ timeout={100}
+ >
+ <div
+ className="source-viewer-measures-card source-viewer-measures-card-fixed-height"
+ >
+ <React.Fragment>
+ <div
+ className="bubble-popup-title"
+ >
+ component_viewer.details
+ </div>
+ <pre>
+ Something failed
+ </pre>
+ <pre />
+ </React.Fragment>
+ </div>
+ </DeferredSpinner>
+</div>
+`;
+
+exports[`should render OK test 1`] = `
+<div
+ className="source-viewer-measures-section source-viewer-measures-section-big js-selected-test"
+>
+ <DeferredSpinner
+ loading={false}
+ timeout={100}
+ >
+ <div
+ className="source-viewer-measures-card source-viewer-measures-card-fixed-height"
+ >
+ <React.Fragment>
+ <div
+ className="bubble-popup-title"
+ >
+ component_viewer.transition.covers
+ </div>
+ <div
+ className="bubble-popup-section"
+ key="project:src/file.js"
+ >
+ <Link
+ onlyActiveOnIndex={false}
+ style={Object {}}
+ to={
+ Object {
+ "pathname": "/dashboard",
+ "query": Object {
+ "branch": undefined,
+ "id": "project:src/file.js",
+ },
+ }
+ }
+ >
+ src/file.js
+ </Link>
+ <span
+ className="note spacer-left"
+ >
+ component_viewer.x_lines_are_covered.3
+ </span>
+ </div>
+ </React.Fragment>
+ </div>
+ </DeferredSpinner>
+</div>
+`;
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayMeasure-test.tsx.snap b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayMeasure-test.tsx.snap
new file mode 100644
index 00000000000..4612ee4034a
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayMeasure-test.tsx.snap
@@ -0,0 +1,53 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`should render 1`] = `
+<div
+ className="measure measure-one-line"
+ data-metric="coverage"
+ key="coverage"
+>
+ <span
+ className="measure-name"
+ >
+ Coverage
+ </span>
+ <span
+ className="measure-value"
+ >
+ <Measure
+ metricKey="coverage"
+ metricType="PERCENT"
+ small={true}
+ value="72"
+ />
+ </span>
+</div>
+`;
+
+exports[`should render issues icon 1`] = `
+<div
+ className="measure measure-one-line"
+ data-metric="bugs"
+ key="bugs"
+>
+ <span
+ className="measure-name"
+ >
+ <IssueTypeIcon
+ className="little-spacer-right"
+ query="bugs"
+ />
+ Bugs
+ </span>
+ <span
+ className="measure-value"
+ >
+ <Measure
+ metricKey="bugs"
+ metricType="INT"
+ small={true}
+ value="2"
+ />
+ </span>
+</div>
+`;
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayTestCase-test.tsx.snap b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayTestCase-test.tsx.snap
new file mode 100644
index 00000000000..02553e953e9
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayTestCase-test.tsx.snap
@@ -0,0 +1,34 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`should render 1`] = `
+<tr>
+ <td
+ className="source-viewer-test-status"
+ >
+ <TestStatusIcon
+ status="OK"
+ />
+ </td>
+ <td
+ className="source-viewer-test-duration note"
+ >
+ 1ms
+ </td>
+ <td
+ className="source-viewer-test-name"
+ >
+ <a
+ className="js-show-test"
+ href="#"
+ onClick={[Function]}
+ >
+ should work
+ </a>
+ </td>
+ <td
+ className="source-viewer-test-covered-lines note"
+ >
+ 3
+ </td>
+</tr>
+`;
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayTestCases-test.tsx.snap b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayTestCases-test.tsx.snap
new file mode 100644
index 00000000000..78316c348a8
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/SourceViewer/components/__tests__/__snapshots__/MeasuresOverlayTestCases-test.tsx.snap
@@ -0,0 +1,247 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`should render 1`] = `
+<div
+ className="source-viewer-measures"
+>
+ <div
+ className="source-viewer-measures-section source-viewer-measures-section-big"
+ >
+ <div
+ className="source-viewer-measures-card source-viewer-measures-card-fixed-height js-test-list"
+ >
+ <div
+ className="measures"
+ >
+ <table
+ className="source-viewer-tests-list"
+ >
+ <tbody>
+ <tr>
+ <td
+ className="source-viewer-test-status note"
+ colSpan={3}
+ >
+ component_viewer.measure_section.unit_tests
+ <br />
+ <span
+ className="spacer-right"
+ >
+ component_viewer.tests.ordered_by
+ </span>
+ <a
+ className="js-sort-tests-by-duration"
+ data-sort="duration"
+ href="#"
+ onClick={[Function]}
+ >
+ component_viewer.tests.duration
+ </a>
+ <span
+ className="slash-separator"
+ />
+ <a
+ className="js-sort-tests-by-name active-link"
+ data-sort="name"
+ href="#"
+ onClick={[Function]}
+ >
+ component_viewer.tests.test_name
+ </a>
+ <span
+ className="slash-separator"
+ />
+ <a
+ className="js-sort-tests-by-status"
+ data-sort="status"
+ href="#"
+ onClick={[Function]}
+ >
+ component_viewer.tests.status
+ </a>
+ </td>
+ <td
+ className="source-viewer-test-covered-lines note"
+ >
+ component_viewer.covered_lines
+ </td>
+ </tr>
+ <MeasuresOverlayTestCase
+ key="AWGub2mFGZxsAttCZwQx"
+ onClick={[Function]}
+ testCase={
+ Object {
+ "coveredLines": 3,
+ "durationInMs": 8,
+ "fileKey": "test:fake-project-for-tests:src/test/java/bar/SimplestTest.java",
+ "fileName": "src/test/java/bar/SimplestTest.java",
+ "id": "AWGub2mFGZxsAttCZwQx",
+ "name": "testAdd",
+ "status": "OK",
+ }
+ }
+ />
+ <MeasuresOverlayTestCase
+ key="AWGub2mGGZxsAttCZwQz"
+ onClick={[Function]}
+ testCase={
+ Object {
+ "coveredLines": 3,
+ "durationInMs": 2,
+ "fileKey": "test:fake-project-for-tests:src/test/java/bar/SimplestTest.java",
+ "fileName": "src/test/java/bar/SimplestTest.java",
+ "id": "AWGub2mGGZxsAttCZwQz",
+ "name": "testAdd_InError",
+ "status": "ERROR",
+ }
+ }
+ />
+ <MeasuresOverlayTestCase
+ key="AWGub2mGGZxsAttCZwQy"
+ onClick={[Function]}
+ testCase={
+ Object {
+ "coveredLines": 3,
+ "durationInMs": 6,
+ "fileKey": "test:fake-project-for-tests:src/test/java/bar/SimplestTest.java",
+ "fileName": "src/test/java/bar/SimplestTest.java",
+ "id": "AWGub2mGGZxsAttCZwQy",
+ "message": "expected:<9> but was:<2>",
+ "name": "testAdd_WhichFails",
+ "stacktrace": "java.lang.AssertionError: expected:<9> but was:<2>
+ at org.junit.Assert.fail(Assert.java:93)
+ at org.junit.Assert.failNotEquals(Assert.java:647)",
+ "status": "FAILURE",
+ }
+ }
+ />
+ </tbody>
+ </table>
+ </div>
+ </div>
+ </div>
+</div>
+`;
+
+exports[`should render 2`] = `
+<div
+ className="source-viewer-measures"
+>
+ <div
+ className="source-viewer-measures-section source-viewer-measures-section-big"
+ >
+ <div
+ className="source-viewer-measures-card source-viewer-measures-card-fixed-height js-test-list"
+ >
+ <div
+ className="measures"
+ >
+ <table
+ className="source-viewer-tests-list"
+ >
+ <tbody>
+ <tr>
+ <td
+ className="source-viewer-test-status note"
+ colSpan={3}
+ >
+ component_viewer.measure_section.unit_tests
+ <br />
+ <span
+ className="spacer-right"
+ >
+ component_viewer.tests.ordered_by
+ </span>
+ <a
+ className="js-sort-tests-by-duration active-link"
+ data-sort="duration"
+ href="#"
+ onClick={[Function]}
+ >
+ component_viewer.tests.duration
+ </a>
+ <span
+ className="slash-separator"
+ />
+ <a
+ className="js-sort-tests-by-name"
+ data-sort="name"
+ href="#"
+ onClick={[Function]}
+ >
+ component_viewer.tests.test_name
+ </a>
+ <span
+ className="slash-separator"
+ />
+ <a
+ className="js-sort-tests-by-status"
+ data-sort="status"
+ href="#"
+ onClick={[Function]}
+ >
+ component_viewer.tests.status
+ </a>
+ </td>
+ <td
+ className="source-viewer-test-covered-lines note"
+ >
+ component_viewer.covered_lines
+ </td>
+ </tr>
+ <MeasuresOverlayTestCase
+ key="AWGub2mGGZxsAttCZwQz"
+ onClick={[Function]}
+ testCase={
+ Object {
+ "coveredLines": 3,
+ "durationInMs": 2,
+ "fileKey": "test:fake-project-for-tests:src/test/java/bar/SimplestTest.java",
+ "fileName": "src/test/java/bar/SimplestTest.java",
+ "id": "AWGub2mGGZxsAttCZwQz",
+ "name": "testAdd_InError",
+ "status": "ERROR",
+ }
+ }
+ />
+ <MeasuresOverlayTestCase
+ key="AWGub2mGGZxsAttCZwQy"
+ onClick={[Function]}
+ testCase={
+ Object {
+ "coveredLines": 3,
+ "durationInMs": 6,
+ "fileKey": "test:fake-project-for-tests:src/test/java/bar/SimplestTest.java",
+ "fileName": "src/test/java/bar/SimplestTest.java",
+ "id": "AWGub2mGGZxsAttCZwQy",
+ "message": "expected:<9> but was:<2>",
+ "name": "testAdd_WhichFails",
+ "stacktrace": "java.lang.AssertionError: expected:<9> but was:<2>
+ at org.junit.Assert.fail(Assert.java:93)
+ at org.junit.Assert.failNotEquals(Assert.java:647)",
+ "status": "FAILURE",
+ }
+ }
+ />
+ <MeasuresOverlayTestCase
+ key="AWGub2mFGZxsAttCZwQx"
+ onClick={[Function]}
+ testCase={
+ Object {
+ "coveredLines": 3,
+ "durationInMs": 8,
+ "fileKey": "test:fake-project-for-tests:src/test/java/bar/SimplestTest.java",
+ "fileName": "src/test/java/bar/SimplestTest.java",
+ "id": "AWGub2mFGZxsAttCZwQx",
+ "name": "testAdd",
+ "status": "OK",
+ }
+ }
+ />
+ </tbody>
+ </table>
+ </div>
+ </div>
+ </div>
+</div>
+`;
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/styles.css b/server/sonar-web/src/main/js/components/SourceViewer/styles.css
index 2c98827855f..3bcdcdf4415 100644
--- a/server/sonar-web/src/main/js/components/SourceViewer/styles.css
+++ b/server/sonar-web/src/main/js/components/SourceViewer/styles.css
@@ -567,7 +567,6 @@
.measure-big .measure-value {
font-size: 22px;
- font-weight: 300;
}
.measure-big .rating {
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/views/measures-overlay.js b/server/sonar-web/src/main/js/components/SourceViewer/views/measures-overlay.js
deleted file mode 100644
index 925e533b87f..00000000000
--- a/server/sonar-web/src/main/js/components/SourceViewer/views/measures-overlay.js
+++ /dev/null
@@ -1,282 +0,0 @@
-/*
- * 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 $ from 'jquery';
-import { select } from 'd3-selection';
-import { arc as d3Arc, pie as d3Pie } from 'd3-shape';
-import { groupBy, sortBy, toPairs } from 'lodash';
-import Template from './templates/source-viewer-measures.hbs';
-import ModalView from '../../common/modals';
-import { searchIssues } from '../../../api/issues';
-import { getMeasures } from '../../../api/measures';
-import { getAllMetrics } from '../../../api/metrics';
-import { getTests, getCoveredFiles } from '../../../api/tests';
-import * as theme from '../../../app/theme';
-import { getLocalizedMetricName, getLocalizedMetricDomain } from '../../../helpers/l10n';
-import { formatMeasure } from '../../../helpers/measures';
-
-const severityComparator = severity => {
- const SEVERITIES_ORDER = ['BLOCKER', 'CRITICAL', 'MAJOR', 'MINOR', 'INFO'];
- return SEVERITIES_ORDER.indexOf(severity);
-};
-
-export default ModalView.extend({
- template: Template,
- testsOrder: ['ERROR', 'FAILURE', 'OK', 'SKIPPED'],
-
- initialize() {
- this.testsScroll = 0;
- const requests = [this.requestMeasures(), this.requestIssues()];
- if (this.options.component.q === 'UTS') {
- requests.push(this.requestTests());
- }
- Promise.all(requests).then(() => this.render());
- },
-
- events() {
- return {
- ...ModalView.prototype.events.apply(this, arguments),
- 'click .js-sort-tests-by-duration': 'sortTestsByDuration',
- 'click .js-sort-tests-by-name': 'sortTestsByName',
- 'click .js-sort-tests-by-status': 'sortTestsByStatus',
- 'click .js-show-test': 'showTest',
- 'click .js-show-all-measures': 'showAllMeasures'
- };
- },
-
- initPieChart() {
- const trans = function(left, top) {
- return `translate(${left}, ${top})`;
- };
-
- const defaults = {
- size: 40,
- thickness: 8,
- color: '#1f77b4',
- baseColor: theme.barBorderColor
- };
-
- this.$('.js-pie-chart').each(function() {
- const data = [$(this).data('value'), $(this).data('max') - $(this).data('value')];
- const options = { ...defaults, ...$(this).data() };
- const radius = options.size / 2;
-
- const container = select(this);
- const svg = container
- .append('svg')
- .attr('width', options.size)
- .attr('height', options.size);
- const plot = svg.append('g').attr('transform', trans(radius, radius));
- const arc = d3Arc()
- .innerRadius(radius - options.thickness)
- .outerRadius(radius);
- const pie = d3Pie()
- .sort(null)
- .value(d => d);
- const colors = function(i) {
- return i === 0 ? options.color : options.baseColor;
- };
- const sectors = plot.selectAll('path').data(pie(data));
-
- sectors
- .enter()
- .append('path')
- .style('fill', (d, i) => colors(i))
- .attr('d', arc);
- });
- },
-
- onRender() {
- ModalView.prototype.onRender.apply(this, arguments);
- this.initPieChart();
- this.$('.js-test-list').scrollTop(this.testsScroll);
- },
-
- calcAdditionalMeasures(measures) {
- measures.issuesRemediationEffort =
- (Number(measures.sqale_index_raw) || 0) +
- (Number(measures.reliability_remediation_effort_raw) || 0) +
- (Number(measures.security_remediation_effort_raw) || 0);
-
- if (measures.lines_to_cover && measures.uncovered_lines) {
- measures.covered_lines = measures.lines_to_cover_raw - measures.uncovered_lines_raw;
- }
- if (measures.conditions_to_cover && measures.uncovered_conditions) {
- measures.covered_conditions = measures.conditions_to_cover - measures.uncovered_conditions;
- }
- return measures;
- },
-
- prepareMetrics(metrics) {
- metrics = metrics
- .filter(metric => metric.value != null)
- .map(metric => ({ ...metric, name: getLocalizedMetricName(metric) }));
- return sortBy(
- toPairs(groupBy(metrics, 'domain')).map(domain => {
- return {
- name: getLocalizedMetricDomain(domain[0]),
- metrics: domain[1]
- };
- }),
- 'name'
- );
- },
-
- requestMeasures() {
- return getAllMetrics().then(metrics => {
- const metricsToRequest = metrics
- .filter(metric => metric.type !== 'DATA' && !metric.hidden)
- .map(metric => metric.key);
-
- return getMeasures(this.options.component.key, metricsToRequest, this.options.branch).then(
- measures => {
- let nextMeasures = this.options.component.measures || {};
- measures.forEach(measure => {
- const metric = metrics.find(metric => metric.key === measure.metric);
- nextMeasures[metric.key] = formatMeasure(measure.value, metric.type);
- nextMeasures[metric.key + '_raw'] = measure.value;
- metric.value = nextMeasures[metric.key];
- });
- nextMeasures = this.calcAdditionalMeasures(nextMeasures);
- this.measures = nextMeasures;
- this.measuresToDisplay = this.prepareMetrics(metrics);
- },
- () => {}
- );
- });
- },
-
- requestIssues() {
- const options = {
- branch: this.options.branch,
- componentKeys: this.options.component.key,
- resolved: false,
- ps: 1,
- facets: 'types,severities,tags'
- };
-
- return searchIssues(options).then(
- data => {
- const typesFacet = data.facets.find(facet => facet.property === 'types').values;
- const typesOrder = ['BUG', 'VULNERABILITY', 'CODE_SMELL'];
- const sortedTypesFacet = sortBy(typesFacet, v => typesOrder.indexOf(v.val));
-
- const severitiesFacet = data.facets.find(facet => facet.property === 'severities').values;
- const sortedSeveritiesFacet = sortBy(severitiesFacet, facet =>
- severityComparator(facet.val)
- );
-
- const tagsFacet = data.facets.find(facet => facet.property === 'tags').values;
-
- this.tagsFacet = tagsFacet;
- this.typesFacet = sortedTypesFacet;
- this.severitiesFacet = sortedSeveritiesFacet;
- this.issuesCount = data.total;
- },
- () => {}
- );
- },
-
- requestTests() {
- return getTests({ branch: this.options.branch, testFileKey: this.options.component.key }).then(
- data => {
- this.tests = data.tests;
- this.testSorting = 'status';
- this.testAsc = true;
- this.sortTests(test => `${this.testsOrder.indexOf(test.status)}_______${test.name}`);
- },
- () => {}
- );
- },
-
- sortTests(condition) {
- let tests = this.tests;
- if (Array.isArray(tests)) {
- tests = sortBy(tests, condition);
- if (!this.testAsc) {
- tests.reverse();
- }
- this.tests = tests;
- }
- },
-
- sortTestsByDuration() {
- if (this.testSorting === 'duration') {
- this.testAsc = !this.testAsc;
- }
- this.sortTests('durationInMs');
- this.testSorting = 'duration';
- this.render();
- },
-
- sortTestsByName() {
- if (this.testSorting === 'name') {
- this.testAsc = !this.testAsc;
- }
- this.sortTests('name');
- this.testSorting = 'name';
- this.render();
- },
-
- sortTestsByStatus() {
- if (this.testSorting === 'status') {
- this.testAsc = !this.testAsc;
- }
- this.sortTests(test => `${this.testsOrder.indexOf(test.status)}_______${test.name}`);
- this.testSorting = 'status';
- this.render();
- },
-
- showTest(e) {
- const testId = $(e.currentTarget).data('id');
- this.testsScroll = $(e.currentTarget)
- .scrollParent()
- .scrollTop();
- getCoveredFiles({ testId }).then(
- data => {
- this.coveredFiles = data.files;
- this.selectedTest = this.tests.find(test => test.id === testId);
- this.render();
- },
- () => {}
- );
- },
-
- showAllMeasures() {
- this.$('.js-all-measures').removeClass('hidden');
- this.$('.js-show-all-measures').remove();
- },
-
- serializeData() {
- return {
- ...ModalView.prototype.serializeData.apply(this, arguments),
- ...this.options.component,
- measures: this.measures,
- measuresToDisplay: this.measuresToDisplay,
- tests: this.tests,
- tagsFacet: this.tagsFacet,
- typesFacet: this.typesFacet,
- severitiesFacet: this.severitiesFacet,
- issuesCount: this.issuesCount,
- testSorting: this.testSorting,
- selectedTest: this.selectedTest,
- coveredFiles: this.coveredFiles || []
- };
- }
-});
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-all.hbs b/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-all.hbs
deleted file mode 100644
index cac99fea52d..00000000000
--- a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-all.hbs
+++ /dev/null
@@ -1,42 +0,0 @@
-{{#notEmpty measuresToDisplay}}
- <div class="source-viewer-measures-section source-viewer-measures-section-big">
- {{#eachEven measuresToDisplay}}
- <div class="source-viewer-measures-card">
- <div class="measures">
- <div class="measures-list">
- <div class="measure measure-one-line measure-big">
- <span class="measure-name">{{name}}</span>
- </div>
- {{#each metrics}}
- <div class="measure measure-one-line" data-metric="{{key}}">
- <span class="measure-name">{{#eq key 'bugs'}}{{issueTypeIcon 'BUG'}} {{/eq}}{{#eq key 'vulnerabilities'}}{{issueTypeIcon 'VULNERABILITY'}} {{/eq}}{{#eq key 'code_smells'}}{{issueTypeIcon 'CODE_SMELL'}} {{/eq}}{{name}}</span>
- <span class="measure-value">&nbsp;{{value}}</span>
- </div>
- {{/each}}
- </div>
- </div>
- </div>
- {{/eachEven}}
- </div>
-
- <div class="source-viewer-measures-section source-viewer-measures-section-big">
- {{#eachOdd measuresToDisplay}}
- <div class="source-viewer-measures-card">
- <div class="measures">
- <div class="measures-list">
- <div class="measure measure-one-line measure-big">
- <span class="measure-name">{{name}}</span>
- </div>
- {{#each metrics}}
- <div class="measure measure-one-line" data-metric="{{key}}">
- <span class="measure-name">{{#eq key 'bugs'}}{{issueTypeIcon 'BUG'}} {{/eq}}{{#eq key 'vulnerabilities'}}{{issueTypeIcon 'VULNERABILITY'}} {{/eq}}{{#eq key 'code_smells'}}{{issueTypeIcon 'CODE_SMELL'}} {{/eq}}{{name}}</span>
- <span class="measure-value">&nbsp;{{value}}</span>
- </div>
- {{/each}}
- </div>
- </div>
- </div>
- {{/eachOdd}}
- </div>
-{{/notEmpty}}
-
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-coverage.hbs b/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-coverage.hbs
deleted file mode 100644
index 2598e962d77..00000000000
--- a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-coverage.hbs
+++ /dev/null
@@ -1,36 +0,0 @@
-{{#if measures.coverage}}
- <div class="measures">
- <div class="measures-chart">
- <span class="js-pie-chart"
- data-value="{{measures.coverage_raw}}"
- data-max="100"
- data-color="#00aa00"
- data-base-color="#d4333f"
- data-size="47"></span>
- </div>
- <div class="measure measure-big" data-metric="coverage">
- <span class="measure-value">{{measures.coverage}}</span>
- <span class="measure-name">{{t 'metric.coverage.name'}}</span>
- </div>
- </div>
-
- {{#any measures.covered_lines measures.lines_to_cover measures.covered_conditions measures.conditions_to_cover}}
- <div class="measures">
- <div class="measures-list">
- <div class="measure measure-one-line">
- <span class="measure-name">Covered by Tests</span>
- </div>
- <div class="measure measure-one-line" data-metric="lines_to_cover">
- <span class="measure-name">Lines</span>
- <span class="measure-value">{{formatMeasure measures.covered_lines 'INT'}}/{{measures.lines_to_cover}}</span>
- </div>
- {{#if measures.conditions_to_cover}}
- <div class="measure measure-one-line" data-metric="conditions_to_cover">
- <span class="measure-name">Conditions</span>
- <span class="measure-value">{{default measures.covered_conditions 0}}/{{measures.conditions_to_cover}}</span>
- </div>
- {{/if}}
- </div>
- </div>
- {{/any}}
-{{/if}}
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-duplications.hbs b/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-duplications.hbs
deleted file mode 100644
index 86ddc9e1edc..00000000000
--- a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-duplications.hbs
+++ /dev/null
@@ -1,30 +0,0 @@
-{{#notNull measures.duplicated_lines_density}}
- <div class="source-viewer-measures-card">
- <div class="measures">
- <div class="measures-chart">
- <span class="js-pie-chart"
- data-value="{{measures.duplicated_lines_density_raw}}"
- data-max="100"
- data-size="50"
- data-color="#797979"></span>
- </div>
- <div class="measure measure-big" data-metric="duplicated_lines_density">
- <span class="measure-value">{{measures.duplicated_lines_density}}</span>
- <span class="measure-name">{{t 'metric.duplicated_lines_density.short_name'}}</span>
- </div>
- </div>
-
- <div class="measures">
- <div class="measures-list">
- <div class="measure measure-one-line" data-metric="duplicated_blocks">
- <span class="measure-name">{{t 'metric.duplicated_blocks.name'}}</span>
- <span class="measure-value">{{measures.duplicated_blocks}}</span>
- </div>
- <div class="measure measure-one-line" data-metric="duplicated_lines">
- <span class="measure-name">{{t 'metric.duplicated_lines.name'}}</span>
- <span class="measure-value">{{measures.duplicated_lines}}</span>
- </div>
- </div>
- </div>
- </div>
-{{/notNull}}
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-issues.hbs b/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-issues.hbs
deleted file mode 100644
index 30a39146750..00000000000
--- a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-issues.hbs
+++ /dev/null
@@ -1,47 +0,0 @@
-<div class="source-viewer-measures-card">
- <div class="measures">
- <div class="measure measure-big" data-metric="violations">
- <span class="measure-value">{{default issuesCount 0}}</span>
- <span class="measure-name">{{t 'metric.violations.name'}}</span>
- </div>
- <div class="measure measure-big" data-metric="sqale_index">
- <span class="measure-value">{{formatMeasure measures.issuesRemediationEffort 'SHORT_WORK_DUR'}}</span>
- <span class="measure-name">{{t 'metric.sqale_index.short_name'}}</span>
- </div>
- </div>
-
- {{#if issuesCount}}
- <div class="measures">
- <div class="measures-list">
- {{#each typesFacet}}
- <div class="measure measure-one-line">
- <span class="measure-name">{{issueTypeIcon val}} {{issueType val}}</span>
- <span class="measure-value">{{formatMeasure count 'SHORT_INT'}}</span>
- </div>
- {{/each}}
- </div>
- </div>
-
- <div class="measures">
- <div class="measures-list">
- {{#each severitiesFacet}}
- <div class="measure measure-one-line">
- <span class="measure-name">{{severityHelper val}}</span>
- <span class="measure-value">{{formatMeasure count 'SHORT_INT'}}</span>
- </div>
- {{/each}}
- </div>
- </div>
-
- <div class="measures">
- <div class="measures-list">
- {{#each tagsFacet}}
- <div class="measure measure-one-line">
- <span class="measure-name"><i class="icon-tags"></i>&nbsp;{{val}}</span>
- <span class="measure-value">{{formatMeasure count 'SHORT_INT'}}</span>
- </div>
- {{/each}}
- </div>
- </div>
- {{/if}}
-</div>
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-lines.hbs b/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-lines.hbs
deleted file mode 100644
index 28464266c79..00000000000
--- a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-lines.hbs
+++ /dev/null
@@ -1,29 +0,0 @@
-<div class="measures">
- <div class="measures-list">
- <div class="measure measure-one-line" data-metric="lines">
- <span class="measure-name">{{t 'metric.lines.name'}}</span>
- <span class="measure-value">{{measures.lines}}</span>
- </div>
- <div class="measure measure-one-line" data-metric="ncloc">
- <span class="measure-name">{{t 'metric.ncloc.name'}}</span>
- <span class="measure-value">{{measures.ncloc}}</span>
- </div>
- <div class="measure measure-one-line" data-metric="comment_lines">
- <span class="measure-name">{{t 'metric.comment_lines_density.short_name'}}</span>
- <span class="measure-value">{{measures.comment_lines_density}} / {{measures.comment_lines}}</span>
- </div>
- </div>
-</div>
-
-<div class="measures">
- <div class="measures-list">
- <div class="measure measure-one-line" data-metric="complexity">
- <span class="measure-name">{{t 'metric.complexity.name'}}</span>
- <span class="measure-value">{{measures.complexity}}</span>
- </div>
- <div class="measure measure-one-line" data-metric="function_complexity">
- <span class="measure-name">{{t 'metric.function_complexity.name'}}</span>
- <span class="measure-value">{{measures.function_complexity}}</span>
- </div>
- </div>
-</div>
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-test-cases.hbs b/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-test-cases.hbs
deleted file mode 100644
index 894ab99d1a7..00000000000
--- a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-test-cases.hbs
+++ /dev/null
@@ -1,72 +0,0 @@
-<div class="source-viewer-measures-section source-viewer-measures-section-big">
- <div class="source-viewer-measures-card source-viewer-measures-card-fixed-height js-test-list">
- <div class="measures">
- <table class="source-viewer-tests-list">
- <tr>
- <td class="source-viewer-test-status note" colspan="3">
- {{t 'component_viewer.measure_section.unit_tests'}}<br>
- {{t 'component_viewer.tests.ordered_by'}}
- <a class="js-sort-tests-by-duration {{#eq testSorting 'duration'}}active-link{{/eq}}">
- {{t 'component_viewer.tests.duration'}}</a>
- /
- <a class="js-sort-tests-by-name {{#eq testSorting 'name'}}active-link{{/eq}}">
- {{t 'component_viewer.tests.test_name'}}</a>
- /
- <a class="js-sort-tests-by-status {{#eq testSorting 'status'}}active-link{{/eq}}">
- {{t 'component_viewer.tests.status'}}</a>
- </td>
- <td class="source-viewer-test-covered-lines note">{{t 'component_viewer.covered_lines'}}</td>
- </tr>
- {{#each tests}}
- <tr>
- {{#eq status 'SKIPPED'}}
- <td class="source-viewer-test-status note">{{testStatusIcon status}}</td>
- <td class="source-viewer-test-duration note"></td>
- <td class="source-viewer-test-name">{{name}}</td>
- <td class="source-viewer-test-covered-lines note"></td>
- {{else}}
- {{#ifTestData this}}
- <td class="source-viewer-test-status note">{{testStatusIcon status}}</td>
- <td class="source-viewer-test-duration note">{{durationInMs}}ms</td>
- <td class="source-viewer-test-name"><a class="js-show-test" data-id="{{id}}">{{name}}</a></td>
- <td class="source-viewer-test-covered-lines note">{{coveredLines}}</td>
- {{else}}
- <td class="source-viewer-test-status note">{{testStatusIcon status}}</td>
- <td class="source-viewer-test-duration note">{{durationInMs}}ms</td>
- <td class="source-viewer-test-name">{{name}}</td>
- {{/ifTestData}}
- {{/eq}}
- </tr>
- {{/each}}
- </table>
- </div>
- </div>
-</div>
-
-{{#if selectedTest}}
- <div class="source-viewer-measures-section source-viewer-measures-section-big js-selected-test">
- <div class="source-viewer-measures-card source-viewer-measures-card-fixed-height">
- {{#notEq selectedTest.status 'ERROR'}}
- {{#notEq selectedTest.status 'FAILURE'}}
- <div class="bubble-popup-title">{{t 'component_viewer.transition.covers'}}</div>
- {{#each coveredFiles}}
- <div class="bubble-popup-section">
- <a target="_blank" href="{{dashboardUrl key}}" title="{{longName}}">{{longName}}</a>
- <span class="note">{{tp 'component_viewer.x_lines_are_covered' coveredLines}}</span>
- </div>
- {{else}}
- {{t 'none'}}
- {{/each}}
- {{/notEq}}
- {{/notEq}}
-
- {{#notEq selectedTest.status 'OK'}}
- <div class="bubble-popup-title">{{t 'component_viewer.details'}}</div>
- {{#if selectedTest.message}}
- <pre>{{selectedTest.message}}</pre>
- {{/if}}
- <pre>{{selectedTest.stacktrace}}</pre>
- {{/notEq}}
- </div>
- </div>
-{{/if}}
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-tests.hbs b/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-tests.hbs
deleted file mode 100644
index c9b33c392ac..00000000000
--- a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/_source-viewer-measures-tests.hbs
+++ /dev/null
@@ -1,40 +0,0 @@
-<div class="source-viewer-measures-card">
- <div class="measures">
- <div class="measures-list">
- <div class="measure measure-big" data-metric="tests">
- <span class="measure-name">{{t 'metric.tests.name'}}</span>
- <span class="measure-value">{{measures.tests}}</span>
- </div>
- {{#notNull measures.test_success_density}}
- <div class="measure measure-one-line" data-metric="test_success_density">
- <span class="measure-name">{{t 'metric.test_success_density.name'}}</span>
- <span class="measure-value">{{measures.test_success_density}}</span>
- </div>
- {{/notNull}}
- {{#notNull measures.test_failures}}
- <div class="measure measure-one-line" data-metric="test_failures">
- <span class="measure-name">{{t 'metric.test_failures.name'}}</span>
- <span class="measure-value">{{measures.test_failures}}</span>
- </div>
- {{/notNull}}
- {{#notNull measures.test_errors}}
- <div class="measure measure-one-line" data-metric="test_errors">
- <span class="measure-name">{{t 'metric.test_errors.name'}}</span>
- <span class="measure-value">{{measures.test_errors}}</span>
- </div>
- {{/notNull}}
- {{#notNull measures.skipped_tests}}
- <div class="measure measure-one-line" data-metric="skipped_tests">
- <span class="measure-name">{{t 'metric.skipped_tests.name'}}</span>
- <span class="measure-value">{{measures.skipped_tests}}</span>
- </div>
- {{/notNull}}
- {{#notNull measures.test_execution_time}}
- <div class="measure measure-one-line" data-metric="test_execution_time">
- <span class="measure-name">{{t 'metric.test_execution_time.name'}}</span>
- <span class="measure-value">{{measures.test_execution_time}}</span>
- </div>
- {{/notNull}}
- </div>
- </div>
-</div>
diff --git a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/source-viewer-measures.hbs b/server/sonar-web/src/main/js/components/SourceViewer/views/templates/source-viewer-measures.hbs
deleted file mode 100644
index ebcc2e79b13..00000000000
--- a/server/sonar-web/src/main/js/components/SourceViewer/views/templates/source-viewer-measures.hbs
+++ /dev/null
@@ -1,68 +0,0 @@
-<div class="modal-container source-viewer-measures-modal">
- <div class="source-viewer-header-component source-viewer-measures-component">
- {{#unless removed}}
- {{#if projectName}}
- <div class="source-viewer-header-component-project">
- {{qualifierIcon 'TRK'}}&nbsp;<a href="{{dashboardUrl project}}">{{projectName}}</a>
- {{#if subProjectName}}
- &nbsp;&nbsp;&nbsp;
- {{qualifierIcon 'TRK'}}&nbsp;<a href="{{dashboardUrl subProject}}">{{subProjectName}}</a>
- {{/if}}
- </div>
- {{/if}}
-
- <div class="source-viewer-header-component-name">
- {{qualifierIcon q}} {{default path longName}}
- </div>
- {{else}}
- <div class="source-viewer-header-component-project removed">{{removedMessage}}</div>
- {{/unless}}
- </div>
-
- {{#eq q 'UTS'}}
- <div class="source-viewer-measures">
- <div class="source-viewer-measures-section">
- {{> '_source-viewer-measures-tests'}}
- </div>
- </div>
- <div class="source-viewer-measures">
- {{> '_source-viewer-measures-test-cases'}}
- </div>
- {{else}}
- <div class="source-viewer-measures">
- <div class="source-viewer-measures-section">
- <div class="source-viewer-measures-card">
- {{> '_source-viewer-measures-lines'}}
- </div>
- </div>
-
- <div class="source-viewer-measures-section">
- {{> '_source-viewer-measures-issues'}}
- </div>
-
- {{#if measures.coverage}}
- <div class="source-viewer-measures-section">
- <div class="source-viewer-measures-card">
- {{> '_source-viewer-measures-coverage'}}
- </div>
- </div>
- {{/if}}
-
- <div class="source-viewer-measures-section">
- {{> '_source-viewer-measures-duplications'}}
- </div>
- </div>
- {{/eq}}
-
-
- <div class="spacer-bottom">&nbsp;</div>
- <a class="js-show-all-measures">{{t 'component_viewer.show_all_measures'}}</a>
-
- <div class="source-viewer-measures source-viewer-measures-secondary js-all-measures hidden">
- {{> '_source-viewer-measures-all'}}
- </div>
-</div>
-
-<div class="modal-foot">
- <a class="js-modal-close" href="#">{{t 'close'}}</a>
-</div>
diff --git a/server/sonar-web/src/main/js/components/measure/Measure.tsx b/server/sonar-web/src/main/js/components/measure/Measure.tsx
index 3437fa9616f..8cccb3cd5d6 100644
--- a/server/sonar-web/src/main/js/components/measure/Measure.tsx
+++ b/server/sonar-web/src/main/js/components/measure/Measure.tsx
@@ -27,18 +27,26 @@ import { formatMeasure } from '../../helpers/measures';
interface Props {
className?: string;
decimals?: number | null;
- value?: string;
metricKey: string;
metricType: string;
+ small?: boolean;
+ value: string | undefined;
}
-export default function Measure({ className, decimals, metricKey, metricType, value }: Props) {
+export default function Measure({
+ className,
+ decimals,
+ metricKey,
+ metricType,
+ small,
+ value
+}: Props) {
if (value === undefined) {
return <span>{'–'}</span>;
}
if (metricType === 'LEVEL') {
- return <Level className={className} level={value} />;
+ return <Level className={className} level={value} small={small} />;
}
if (metricType !== 'RATING') {
@@ -47,7 +55,7 @@ export default function Measure({ className, decimals, metricKey, metricType, va
}
const tooltip = getRatingTooltip(metricKey, Number(value));
- const rating = <Rating value={value} />;
+ const rating = <Rating small={small} value={value} />;
if (tooltip) {
return (
<Tooltips overlay={tooltip}>
diff --git a/server/sonar-web/src/main/js/components/measure/__tests__/Measure-test.tsx b/server/sonar-web/src/main/js/components/measure/__tests__/Measure-test.tsx
index 84aaf15fdb3..cbfe52cb4bf 100644
--- a/server/sonar-web/src/main/js/components/measure/__tests__/Measure-test.tsx
+++ b/server/sonar-web/src/main/js/components/measure/__tests__/Measure-test.tsx
@@ -58,5 +58,7 @@ it('renders unknown RATING', () => {
});
it('renders undefined measure', () => {
- expect(shallow(<Measure metricKey="foo" metricType="PERCENT" />)).toMatchSnapshot();
+ expect(
+ shallow(<Measure metricKey="foo" metricType="PERCENT" value={undefined} />)
+ ).toMatchSnapshot();
});
diff --git a/server/sonar-web/src/main/js/components/shared/TestStatusIcon.tsx b/server/sonar-web/src/main/js/components/shared/TestStatusIcon.tsx
new file mode 100644
index 00000000000..c54fca5654b
--- /dev/null
+++ b/server/sonar-web/src/main/js/components/shared/TestStatusIcon.tsx
@@ -0,0 +1,30 @@
+/*
+ * 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';
+
+interface Props {
+ className?: string;
+ status: string;
+}
+
+export default function TestStatusIcon({ className, status }: Props) {
+ return <i className={classNames('icon-test-status-' + status.toLowerCase(), className)} />;
+}