diff options
author | Stas Vilchik <vilchiks@gmail.com> | 2015-03-10 15:46:31 +0100 |
---|---|---|
committer | Stas Vilchik <vilchiks@gmail.com> | 2015-03-16 12:38:38 +0100 |
commit | 45f9acebd289a81f5e926bd5ca505d68f2a4abd7 (patch) | |
tree | 81a332393171759cf10baa1c526115ea74c5d388 /server/sonar-web/src/main | |
parent | 195c9fc6c44b87d20e084df1d6647e30598298ca (diff) | |
download | sonarqube-45f9acebd289a81f5e926bd5ca505d68f2a4abd7.tar.gz sonarqube-45f9acebd289a81f5e926bd5ca505d68f2a4abd7.zip |
SONAR-6285 add a home page for global issues
Diffstat (limited to 'server/sonar-web/src/main')
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'}} <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'}} <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); +} |