]> source.dussan.org Git - sonarqube.git/commitdiff
SONAR-6285 add a home page for global issues
authorStas Vilchik <vilchiks@gmail.com>
Tue, 10 Mar 2015 14:46:31 +0000 (15:46 +0100)
committerStas Vilchik <vilchiks@gmail.com>
Mon, 16 Mar 2015 11:38:38 +0000 (12:38 +0100)
server/sonar-web/src/main/coffee/issues/controller.coffee
server/sonar-web/src/main/coffee/issues/layout.coffee
server/sonar-web/src/main/coffee/issues/router.coffee
server/sonar-web/src/main/coffee/issues/workspace-home-view.coffee [new file with mode: 0644]
server/sonar-web/src/main/hbs/issues/issues-layout.hbs
server/sonar-web/src/main/hbs/issues/issues-workspace-home.hbs [new file with mode: 0644]
server/sonar-web/src/main/js/graphics/barchart.js
server/sonar-web/src/main/less/components/columns.less
server/sonar-web/src/main/less/init/icons.less
server/sonar-web/src/main/less/pages/issues.less
sonar-core/src/main/resources/org/sonar/l10n/core.properties

index d30e8c7978c02aa51ae556c1e02ffccaf80be2c3..99f39a32afea53f3f11b57205da451a34b804dee 100644 (file)
@@ -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()
+
index 9cef9f326dcf3bb1134400c3d2fe75b82b773c99..e06daa35e74d3cdab86edea60c2b973b3d58aa5d 100644 (file)
@@ -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'
index 4091d9c8decb54c037f9de8e6a6c1ac2dcff1056..ca220549f1f84f637b7e00539418b8378f0601c3 100644 (file)
@@ -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 (file)
index 0000000..704dbb2
--- /dev/null
@@ -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'
index f60fecfb03ffc84eed43ac735bdfd0d13223b8ed..5a61d1f66bf4e097a9ce74c0a67382832f55d06f 100644 (file)
@@ -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 (file)
index 0000000..3889dcd
--- /dev/null
@@ -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>
index e8bd3b7e353646b6a21e75eed0dd12f5fc9ea76f..c107a68661bc215bb7c222b6d2eab03ec1f9e1e3 100644 (file)
@@ -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)
       });
 
                   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) {
index a69f5e25d54986b37c5b8f57e1ca2e282e61d0e4..0e4b489e74b57ed1f40e0cbe126edfce13f09599 100644 (file)
   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);
 }
index f2b19caefdde1a15bff2081fc0ed1cded4584ed6..2c1a113474cb9c6fd4e1ad410efc6219a280fa0d 100644 (file)
@@ -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();
+}
 
 
 /*
index 05c13174fe40fc102b068e0c1763f13622e0e21a..0cc944a4c37d3893db6ba137cfe446feb7333894 100644 (file)
@@ -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;
     }
 
   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);
+}
index 9e3150b7d827e3539e1d49727ed20c4f5910e60d..66a429af3b15259ba91e21c332271a1d04b5e321 100644 (file)
@@ -719,6 +719,13 @@ issue.technical_debt_deleted=Rule not configured to generate technical debt esti
 issue.creation_date=Created
 issues.return_to_list=Return to List
 issues.issues_limit_reached=For usability reasons, only the {0} issues are displayed.
+issues.home.recent_issues=Recent Issues
+issues.home.my_recent_issues=My Recent Issues
+issues.home.over_last_week=Over Last Week
+issues.home.my_filters=My Filters
+issues.home.projects=Projects
+issues.home.authors=Authors
+issues.home.tags=Tags
 
 
 #------------------------------------------------------------------------------