aboutsummaryrefslogtreecommitdiffstats
path: root/server/sonar-web/src/main
diff options
context:
space:
mode:
authorStas Vilchik <vilchiks@gmail.com>2015-03-10 15:46:31 +0100
committerStas Vilchik <vilchiks@gmail.com>2015-03-16 12:38:38 +0100
commit45f9acebd289a81f5e926bd5ca505d68f2a4abd7 (patch)
tree81a332393171759cf10baa1c526115ea74c5d388 /server/sonar-web/src/main
parent195c9fc6c44b87d20e084df1d6647e30598298ca (diff)
downloadsonarqube-45f9acebd289a81f5e926bd5ca505d68f2a4abd7.tar.gz
sonarqube-45f9acebd289a81f5e926bd5ca505d68f2a4abd7.zip
SONAR-6285 add a home page for global issues
Diffstat (limited to 'server/sonar-web/src/main')
-rw-r--r--server/sonar-web/src/main/coffee/issues/controller.coffee22
-rw-r--r--server/sonar-web/src/main/coffee/issues/layout.coffee9
-rw-r--r--server/sonar-web/src/main/coffee/issues/router.coffee9
-rw-r--r--server/sonar-web/src/main/coffee/issues/workspace-home-view.coffee146
-rw-r--r--server/sonar-web/src/main/hbs/issues/issues-layout.hbs1
-rw-r--r--server/sonar-web/src/main/hbs/issues/issues-workspace-home.hbs77
-rw-r--r--server/sonar-web/src/main/js/graphics/barchart.js4
-rw-r--r--server/sonar-web/src/main/less/components/columns.less9
-rw-r--r--server/sonar-web/src/main/less/init/icons.less5
-rw-r--r--server/sonar-web/src/main/less/pages/issues.less20
10 files changed, 296 insertions, 6 deletions
diff --git a/server/sonar-web/src/main/coffee/issues/controller.coffee b/server/sonar-web/src/main/coffee/issues/controller.coffee
index d30e8c7978c..99f39a32afe 100644
--- a/server/sonar-web/src/main/coffee/issues/controller.coffee
+++ b/server/sonar-web/src/main/coffee/issues/controller.coffee
@@ -22,10 +22,12 @@ define [
'components/navigator/controller'
'issues/component-viewer/main'
+ 'issues/workspace-home-view'
], (
Controller
ComponentViewer
+ HomeView
) ->
$ = jQuery
@@ -61,6 +63,7 @@ define [
fetchList: (firstPage = true) ->
if firstPage
@options.app.state.set { selectedIndex: 0, page: 1 }, { silent: true }
+ @hideHomePage()
@closeComponentViewer()
data = @_issuesParameters()
@@ -201,3 +204,22 @@ define [
@options.app.issuesView.bindScrollEvents()
@options.app.issuesView.scrollTo()
+
+ showHomePage: ->
+ @fetchList()
+ @options.app.layout.workspaceComponentViewerRegion.reset()
+ key.setScope 'home'
+ @options.app.issuesView.unbindScrollEvents()
+ @options.app.homeView = new HomeView app: @options.app
+ @options.app.layout.workspaceHomeRegion.show @options.app.homeView
+ @options.app.layout.showHomePage()
+
+
+ hideHomePage: ->
+ @options.app.layout.workspaceComponentViewerRegion.reset()
+ @options.app.layout.workspaceHomeRegion.reset()
+ key.setScope 'list'
+ @options.app.layout.hideHomePage()
+ @options.app.issuesView.bindScrollEvents()
+ @options.app.issuesView.scrollTo()
+
diff --git a/server/sonar-web/src/main/coffee/issues/layout.coffee b/server/sonar-web/src/main/coffee/issues/layout.coffee
index 9cef9f326dc..e06daa35e74 100644
--- a/server/sonar-web/src/main/coffee/issues/layout.coffee
+++ b/server/sonar-web/src/main/coffee/issues/layout.coffee
@@ -35,6 +35,7 @@ define [
workspaceHeaderRegion: '.search-navigator-workspace-header'
workspaceListRegion: '.search-navigator-workspace-list'
workspaceComponentViewerRegion: '.issues-workspace-component-viewer'
+ workspaceHomeRegion: '.issues-workspace-home'
onRender: ->
@@ -58,3 +59,11 @@ define [
hideComponentViewer: ->
$('.issues').removeClass 'issues-extended-view'
$(window).scrollTop @scroll if @scroll?
+
+
+ showHomePage: ->
+ $('.issues').addClass 'issues-home-view'
+
+
+ hideHomePage: ->
+ $('.issues').removeClass 'issues-home-view'
diff --git a/server/sonar-web/src/main/coffee/issues/router.coffee b/server/sonar-web/src/main/coffee/issues/router.coffee
index 4091d9c8dec..ca220549f1f 100644
--- a/server/sonar-web/src/main/coffee/issues/router.coffee
+++ b/server/sonar-web/src/main/coffee/issues/router.coffee
@@ -26,7 +26,7 @@ define [
class extends Router
routes:
- '': 'emptyQuery'
+ '': 'home'
':query': 'index'
@@ -35,8 +35,11 @@ define [
@listenTo options.app.state, 'change:filter', @updateRoute
- emptyQuery: ->
- @navigate 'resolved=false', { trigger: true, replace: true }
+ home: ->
+ if @options.app.state.get 'isContext'
+ @navigate 'resolved=false', { trigger: true, replace: true }
+ else
+ @options.app.controller.showHomePage()
index: (query) ->
diff --git a/server/sonar-web/src/main/coffee/issues/workspace-home-view.coffee b/server/sonar-web/src/main/coffee/issues/workspace-home-view.coffee
new file mode 100644
index 00000000000..704dbb28834
--- /dev/null
+++ b/server/sonar-web/src/main/coffee/issues/workspace-home-view.coffee
@@ -0,0 +1,146 @@
+#
+# SonarQube, open source software quality management tool.
+# Copyright (C) 2008-2014 SonarSource
+# mailto:contact AT sonarsource DOT com
+#
+# SonarQube 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.
+#
+# SonarQube 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.
+#
+
+define [
+ 'widgets/issue-filter'
+ 'templates/issues'
+], (IssueFilter) ->
+
+ $ = jQuery
+
+
+ Handlebars.registerHelper 'issuesHomeLink', (property, value) ->
+ "#{baseUrl}/issues/search#resolved=false|createdInLast=1w|#{property}=#{encodeURIComponent value}"
+
+ Handlebars.registerHelper 'myIssuesHomeLink', (property, value) ->
+ "#{baseUrl}/issues/search#resolved=false|createdInLast=1w|assignees=__me__|#{property}=#{encodeURIComponent value}"
+
+ Handlebars.registerHelper 'issueFilterHomeLink', (id) ->
+ "#{baseUrl}/issues/search#id=#{id}"
+
+
+ class extends Marionette.ItemView
+ template: Templates['issues-workspace-home']
+
+
+ modelEvents:
+ 'change': 'render'
+
+
+ events:
+ 'click .js-barchart rect': 'selectBar'
+ 'click .js-my-barchart rect': 'selectMyBar'
+
+
+ initialize: ->
+ @model = new Backbone.Model
+ @requestIssues()
+ @requestMyIssues()
+
+
+ _getProjects: (r) ->
+ projectFacet = _.findWhere r.facets, property: 'projectUuids'
+ values = _.head projectFacet.values, 3
+ values.forEach (v) =>
+ project = _.findWhere r.projects, uuid: v.val
+ v.label = project.longName
+ values
+
+
+ _getAuthors: (r) ->
+ authorFacet = _.findWhere r.facets, property: 'authors'
+ values = _.head authorFacet.values, 3
+ values
+
+
+ _getTags: (r) ->
+ MIN_SIZE = 10
+ MAX_SIZE = 24
+ tagFacet = _.findWhere r.facets, property: 'tags'
+ values = _.head tagFacet.values, 10
+ minCount = _.min(values, (v) -> v.count).count
+ maxCount = _.max(values, (v) -> v.count).count
+ scale = d3.scale.linear().domain([minCount, maxCount]).range([MIN_SIZE, MAX_SIZE])
+ values.forEach (v) =>
+ v.size = scale v.count
+ values
+
+
+ requestIssues: ->
+ url = "#{baseUrl}/api/issues/search"
+ options =
+ resolved: false
+ createdInLast: '1w'
+ ps: 1
+ facets: 'createdAt,projectUuids,authors,tags'
+ $.get(url, options).done (r) =>
+ @model.set
+ createdAt: _.findWhere(r.facets, property: 'createdAt').values
+ projects: @_getProjects r
+ authors: @_getAuthors r
+ tags: @_getTags r
+
+
+ requestMyIssues: ->
+ url = "#{baseUrl}/api/issues/search"
+ options =
+ resolved: false
+ createdInLast: '1w'
+ assignees: '__me__'
+ ps: 1
+ facets: 'createdAt,projectUuids,authors,tags'
+ $.get(url, options).done (r) =>
+ @model.set
+ myCreatedAt: _.findWhere(r.facets, property: 'createdAt').values
+ myProjects: @_getProjects r
+ myTags: @_getTags r
+
+
+ onRender: ->
+ values = @model.get 'createdAt'
+ myValues = @model.get 'myCreatedAt'
+ @$('.js-barchart').barchart values if values?
+ @$('.js-my-barchart').barchart myValues if myValues?
+ @$('[data-toggle="tooltip"]').tooltip container: 'body'
+
+
+ selectBar: (e) ->
+ periodStart = $(e.currentTarget).data 'period-start'
+ periodEnd = $(e.currentTarget).data 'period-end'
+ @options.app.state.setQuery
+ resolved: false
+ createdAfter: periodStart
+ createdBefore: periodEnd
+
+
+ selectMyBar: (e) ->
+ periodStart = $(e.currentTarget).data 'period-start'
+ periodEnd = $(e.currentTarget).data 'period-end'
+ @options.app.state.setQuery
+ resolved: false
+ assignees: '__me__'
+ createdAfter: periodStart
+ createdBefore: periodEnd
+
+
+ serializeData: ->
+ _.extend super,
+ user: window.SS.user
+ filters: _.sortBy @options.app.filters.toJSON(), 'name'
diff --git a/server/sonar-web/src/main/hbs/issues/issues-layout.hbs b/server/sonar-web/src/main/hbs/issues/issues-layout.hbs
index f60fecfb03f..5a61d1f66bf 100644
--- a/server/sonar-web/src/main/hbs/issues/issues-layout.hbs
+++ b/server/sonar-web/src/main/hbs/issues/issues-layout.hbs
@@ -7,4 +7,5 @@
<div class="search-navigator-workspace-header"></div>
<div class="search-navigator-workspace-list"></div>
<div class="issues-workspace-component-viewer"></div>
+ <div class="issues-workspace-home"></div>
</div>
diff --git a/server/sonar-web/src/main/hbs/issues/issues-workspace-home.hbs b/server/sonar-web/src/main/hbs/issues/issues-workspace-home.hbs
new file mode 100644
index 00000000000..3889dcdd978
--- /dev/null
+++ b/server/sonar-web/src/main/hbs/issues/issues-workspace-home.hbs
@@ -0,0 +1,77 @@
+<div class="spacer-top spacer-bottom">
+ <div class="columns">
+ <div class="column-half {{#unless user}}column-one{{/unless}}">
+ <h3 class="text-center">{{t 'issues.home.recent_issues'}}</h3>
+ <p class="note text-center">({{t 'issues.home.over_last_week'}})</p>
+
+ <div class="spacer-top text-center js-barchart" data-height="75" data-width="300"></div>
+ <h4 class="spacer-top spacer-bottom text-center">{{t 'issues.home.projects'}}</h4>
+ <table class="data zebra spacer-top">
+ {{#each projects}}
+ <tr>
+ <td>{{qualifierIcon 'TRK'}}&nbsp;<a href="{{issuesHomeLink 'projectUuids' val}}">{{label}}</a></td>
+ <td class="thin text-right">+{{numberShort count}}</td>
+ </tr>
+ {{/each}}
+ </table>
+ <h4 class="spacer-top spacer-bottom text-center">{{t 'issues.home.authors'}}</h4>
+ <table class="data zebra spacer-top">
+ {{#each authors}}
+ <tr>
+ <td><a href="{{issuesHomeLink 'authors' val}}">{{val}}</a></td>
+ <td class="thin text-right">+{{numberShort count}}</td>
+ </tr>
+ {{/each}}
+ </table>
+ <h4 class="spacer-top spacer-bottom text-center">{{t 'issues.home.tags'}}</h4>
+ <ul class="list-inline">
+ {{#each tags}}
+ <li><a class="link-no-underline" style="font-size: {{size}}px;" data-toggle="tooltip" data-placement="bottom"
+ href="{{issuesHomeLink 'tags' val}}"
+ title="+{{numberShort count}}">{{val}}</a></li>
+ {{/each}}
+ </ul>
+ </div>
+
+ {{#if user}}
+ <div class="column-half">
+ <h3 class="text-center">{{t 'issues.home.my_recent_issues'}}</h3>
+ <p class="note text-center">({{t 'issues.home.over_last_week'}})</p>
+
+ <div class="spacer-top text-center js-my-barchart" data-height="75" data-width="300"></div>
+ {{#notEmpty myProjects}}
+ <h4 class="spacer-top spacer-bottom text-center">{{t 'issues.home.projects'}}</h4>
+ <table class="data zebra spacer-top">
+ {{#each myProjects}}
+ <tr>
+ <td>{{qualifierIcon 'TRK'}}&nbsp;<a href="{{myIssuesHomeLink 'projectUuids' val}}">{{label}}</a></td>
+ <td class="thin text-right">+{{numberShort count}}</td>
+ </tr>
+ {{/each}}
+ </table>
+ {{/notEmpty}}
+ {{#notEmpty myTags}}
+ <h4 class="spacer-top spacer-bottom text-center">{{t 'issues.home.tags'}}</h4>
+ <ul class="list-inline">
+ {{#each myTags}}
+ <li><a class="link-no-underline" style="font-size: {{size}}px;" data-toggle="tooltip" data-placement="bottom"
+ href="{{myIssuesHomeLink 'tags' val}}"
+ title="+{{numberShort count}}">{{val}}</a></li>
+ {{/each}}
+ </ul>
+ {{/notEmpty}}
+ {{#notEmpty filters}}
+ <h3 class="spacer-bottom text-center" style="padding-top: 12px; margin-top: 20px; border-top: 1px solid #efefef;">
+ {{t 'issues.home.my_filters'}}</h3>
+ <ul class="list-inline">
+ {{#each filters}}
+ <li>
+ <a href="{{issueFilterHomeLink id}}">{{name}}</a>
+ </li>
+ {{/each}}
+ </ul>
+ {{/notEmpty}}
+ </div>
+ {{/if}}
+ </div>
+</div>
diff --git a/server/sonar-web/src/main/js/graphics/barchart.js b/server/sonar-web/src/main/js/graphics/barchart.js
index e8bd3b7e353..c107a68661b 100644
--- a/server/sonar-web/src/main/js/graphics/barchart.js
+++ b/server/sonar-web/src/main/js/graphics/barchart.js
@@ -48,7 +48,7 @@
$(this).each(function () {
var options = _.defaults($(this).data(), defaults());
_.extend(options, {
- width: $(this).width(),
+ width: options.width || $(this).width(),
endDate: moment(options.endDate)
});
@@ -116,7 +116,7 @@
isSameDay = ending.diff(beginning, 'days') <= 1;
return d.count + '<br>' + beginning.format('LL') + (isSameDay ? '' : (' – ' + ending.format('LL')));
})
- .attr('data-placement', 'right')
+ .attr('data-placement', 'bottom')
.attr('data-toggle', 'tooltip');
var maxValue = d3.max(data, function (d) {
diff --git a/server/sonar-web/src/main/less/components/columns.less b/server/sonar-web/src/main/less/components/columns.less
index a69f5e25d54..0e4b489e74b 100644
--- a/server/sonar-web/src/main/less/components/columns.less
+++ b/server/sonar-web/src/main/less/components/columns.less
@@ -31,4 +31,13 @@
width: 50%;
padding: 0 10px;
.box-sizing(border-box);
+
+ &.column-one { margin: 0 25%; }
+}
+
+.column-third {
+ float: left;
+ width: 33%;
+ padding: 0 10px;
+ .box-sizing(border-box);
}
diff --git a/server/sonar-web/src/main/less/init/icons.less b/server/sonar-web/src/main/less/init/icons.less
index f2b19caefdd..2c1a113474c 100644
--- a/server/sonar-web/src/main/less/init/icons.less
+++ b/server/sonar-web/src/main/less/init/icons.less
@@ -581,6 +581,11 @@ a[class^="icon-"], a[class*=" icon-"] {
content: "\f068";
font-size: @iconSmallFontSize;
}
+.icon-issues {
+ display: inline-block;
+ .square(60px);
+ background-image: url();
+}
/*
diff --git a/server/sonar-web/src/main/less/pages/issues.less b/server/sonar-web/src/main/less/pages/issues.less
index 05c13174fe4..0cc944a4c37 100644
--- a/server/sonar-web/src/main/less/pages/issues.less
+++ b/server/sonar-web/src/main/less/pages/issues.less
@@ -29,7 +29,8 @@
&.sticky {
.issues-workspace-list,
- .issues-workspace-component-viewer {
+ .issues-workspace-component-viewer,
+ .issues-workspace-home {
padding-top: 22px + 5px + 5px + 1px + 10px;
}
@@ -153,3 +154,20 @@
background-size: 4px;
background-position: bottom;
}
+
+.issues-home-view {
+ .search-navigator-workspace-list,
+ .issues-workspace-component-viewer {
+ display: none;
+ }
+}
+
+.issues-workspace-home {
+ width: ~"calc(100vw - @{sideWidth})";
+ max-width: 900px;
+ margin-left: auto;
+ margin-right: auto;
+ padding-left: 10px;
+ padding-right: 10px;
+ .box-sizing(border-box);
+}