aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/CorePlugin.java52
-rw-r--r--plugins/sonar-l10n-en-plugin/src/main/resources/org/sonar/l10n/core.properties10
-rw-r--r--sonar-plugin-api/src/main/java/org/sonar/api/Property.java5
-rw-r--r--sonar-server/src/main/webapp/WEB-INF/app/controllers/project_controller.rb13
-rw-r--r--sonar-server/src/main/webapp/WEB-INF/app/controllers/settings_controller.rb43
-rw-r--r--sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_delete_project.html.erb2
-rw-r--r--sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_exclusions.html.erb2
-rw-r--r--sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_links.html.erb2
-rw-r--r--sonar-server/src/main/webapp/WEB-INF/app/views/settings/_plugins.html.erb227
9 files changed, 207 insertions, 149 deletions
diff --git a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/CorePlugin.java b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/CorePlugin.java
index 60f7d250ba0..db162da7c70 100644
--- a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/CorePlugin.java
+++ b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/CorePlugin.java
@@ -48,21 +48,25 @@ import java.util.List;
key = CoreProperties.ORGANIZATION,
name = "Organization",
description = "Identify your installation. Required to generate the server key and to benefit from licensed plugins. Server must be restarted for the change to take effect.",
- global = true),
+ global = true,
+ category = "General"
+ ),
@Property(
key = CoreProperties.SERVER_BASE_URL,
defaultValue = CoreProperties.SERVER_BASE_URL_DEFAULT_VALUE,
name = "Server base URL",
description = "HTTP address of the Sonar server, such as <i>http://yourhost.yourdomain/sonar</i>. This value is used i.e. to create links in emails and to generate server key.",
project = false,
- global = true),
+ global = true,
+ category = "General"),
@Property(
key = CoreProperties.CORE_COVERAGE_PLUGIN_PROPERTY,
defaultValue = "cobertura",
name = "Code coverage plugin",
description = "Key of the code coverage plugin to use.",
project = true,
- global = true),
+ global = true,
+ category = "Coverage"),
@Property(
key = CoreProperties.CORE_IMPORT_SOURCES_PROPERTY,
defaultValue = "" + CoreProperties.CORE_IMPORT_SOURCES_DEFAULT_VALUE,
@@ -70,14 +74,16 @@ import java.util.List;
description = "Set to false if sources should not be displayed, e.g. for security reasons.",
project = true,
module = true,
- global = true),
+ global = true,
+ category = "General"),
@Property(
key = CoreProperties.CORE_TENDENCY_DEPTH_PROPERTY,
defaultValue = "" + CoreProperties.CORE_TENDENCY_DEPTH_DEFAULT_VALUE,
name = "Tendency period",
description = TendencyDecorator.PROP_DAYS_DESCRIPTION,
project = false,
- global = true),
+ global = true,
+ category = "General"),
@Property(
key = CoreProperties.SKIP_TENDENCIES_PROPERTY,
defaultValue = "" + CoreProperties.SKIP_TENDENCIES_DEFAULT_VALUE,
@@ -85,41 +91,47 @@ import java.util.List;
description = "Skip calculation of measure tendencies",
project = true,
module = false,
- global = true),
+ global = true,
+ category = "General"),
@Property(
key = CoreProperties.CORE_SKIPPED_MODULES_PROPERTY,
name = "Exclude modules",
description = "Maven artifact ids of modules to exclude (comma-separated).",
project = true,
- global = false),
+ global = false,
+ category = "General"),
@Property(
key = CoreProperties.CORE_RULE_WEIGHTS_PROPERTY,
defaultValue = CoreProperties.CORE_RULE_WEIGHTS_DEFAULT_VALUE,
name = "Rules weight",
description = "A weight is associated to each priority to calculate the Rules Compliance Index.",
project = false,
- global = true),
+ global = true,
+ category = "General"),
@Property(
key = CoreProperties.CORE_FORCE_AUTHENTICATION_PROPERTY,
defaultValue = "" + CoreProperties.CORE_FORCE_AUTHENTICATION_DEFAULT_VALUE,
name = "Force user authentication",
description = "Forcing user authentication stops un-logged users to access Sonar.",
project = false,
- global = true),
+ global = true,
+ category = "General"),
@Property(
key = CoreProperties.CORE_ALLOW_USERS_TO_SIGNUP_PROPERTY,
defaultValue = "" + CoreProperties.CORE_ALLOW_USERS_TO_SIGNUP_DEAULT_VALUE,
name = "Allow users to sign up online",
description = "Users can sign up online.",
project = false,
- global = true),
+ global = true,
+ category = "General"),
@Property(
key = CoreProperties.CORE_DEFAULT_GROUP,
defaultValue = CoreProperties.CORE_DEFAULT_GROUP_DEFAULT_VALUE,
name = "Default user group",
description = "Any new users will automatically join this group.",
project = false,
- global = true
+ global = true,
+ category = "General"
),
@Property(
key = CoreProperties.CORE_VIOLATION_LOCALE_PROPERTY,
@@ -127,7 +139,8 @@ import java.util.List;
name = "Locale used for violation messages",
description = "Locale to be used when generating violation messages. It's up to each rule engine to support this global internationalization property",
project = true,
- global = true),
+ global = true,
+ category = "General"),
@Property(
key = "sonar.timemachine.period1",
name = "Period 1",
@@ -136,7 +149,8 @@ import java.util.List;
"compare to previous analysis</li><li>A version, for example 1.2</li></ul>",
project = false,
global = true,
- defaultValue = CoreProperties.TIMEMACHINE_DEFAULT_PERIOD_1
+ defaultValue = CoreProperties.TIMEMACHINE_DEFAULT_PERIOD_1,
+ category = "Time Machine"
),
@Property(
key = "sonar.timemachine.period2",
@@ -144,7 +158,8 @@ import java.util.List;
description = "See the property 'Period 1'",
project = false,
global = true,
- defaultValue = CoreProperties.TIMEMACHINE_DEFAULT_PERIOD_2
+ defaultValue = CoreProperties.TIMEMACHINE_DEFAULT_PERIOD_2,
+ category = "Time Machine"
),
@Property(
key = "sonar.timemachine.period3",
@@ -152,7 +167,8 @@ import java.util.List;
description = "See the property 'Period 1'",
project = false,
global = true,
- defaultValue = CoreProperties.TIMEMACHINE_DEFAULT_PERIOD_3
+ defaultValue = CoreProperties.TIMEMACHINE_DEFAULT_PERIOD_3,
+ category = "Time Machine"
),
@Property(
key = "sonar.timemachine.period4",
@@ -162,7 +178,8 @@ import java.util.List;
"for example 2010-12-25</li><li>'previous_analysis' to compare to previous analysis</li><li>A version, for example 1.2</li></ul>",
project = true,
global = false,
- defaultValue = CoreProperties.TIMEMACHINE_DEFAULT_PERIOD_4
+ defaultValue = CoreProperties.TIMEMACHINE_DEFAULT_PERIOD_4,
+ category = "Time Machine"
),
@Property(
key = "sonar.timemachine.period5",
@@ -170,7 +187,8 @@ import java.util.List;
description = "See the property 'Period 4'",
project = true,
global = false,
- defaultValue = CoreProperties.TIMEMACHINE_DEFAULT_PERIOD_5
+ defaultValue = CoreProperties.TIMEMACHINE_DEFAULT_PERIOD_5,
+ category = "Time Machine"
)
})
public class CorePlugin extends SonarPlugin {
diff --git a/plugins/sonar-l10n-en-plugin/src/main/resources/org/sonar/l10n/core.properties b/plugins/sonar-l10n-en-plugin/src/main/resources/org/sonar/l10n/core.properties
index 6695634552d..4d5541356c2 100644
--- a/plugins/sonar-l10n-en-plugin/src/main/resources/org/sonar/l10n/core.properties
+++ b/plugins/sonar-l10n-en-plugin/src/main/resources/org/sonar/l10n/core.properties
@@ -416,6 +416,16 @@ dashboard.update_dashboard=Update dashboard
#------------------------------------------------------------------------------
#
+# SETTINGS
+#
+#------------------------------------------------------------------------------
+settings.save_category=Save {0} settings
+settings.category.General=General
+
+
+
+#------------------------------------------------------------------------------
+#
# WIDGETS
#
#------------------------------------------------------------------------------
diff --git a/sonar-plugin-api/src/main/java/org/sonar/api/Property.java b/sonar-plugin-api/src/main/java/org/sonar/api/Property.java
index b71fe2af2d4..02f38d19331 100644
--- a/sonar-plugin-api/src/main/java/org/sonar/api/Property.java
+++ b/sonar-plugin-api/src/main/java/org/sonar/api/Property.java
@@ -58,6 +58,11 @@ public @interface Property {
String description() default "";
/**
+ * @since 2.11
+ */
+ String category() default "";
+
+ /**
* Is the property displayed in projet settings page ?
*/
boolean project() default false;
diff --git a/sonar-server/src/main/webapp/WEB-INF/app/controllers/project_controller.rb b/sonar-server/src/main/webapp/WEB-INF/app/controllers/project_controller.rb
index 79ae8f0ed59..d5c5bceae45 100644
--- a/sonar-server/src/main/webapp/WEB-INF/app/controllers/project_controller.rb
+++ b/sonar-server/src/main/webapp/WEB-INF/app/controllers/project_controller.rb
@@ -71,6 +71,19 @@ class ProjectController < ApplicationController
if !@project.project? && !@project.module?
redirect_to :action => 'index', :id => params[:id]
end
+
+ @category=params[:category] ||= 'General'
+ @properties_per_category={}
+ java_facade.getPluginsMetadata().each do |plugin|
+ properties=java_facade.getPluginProperties(plugin).select { |property|
+ (@project.module? && property.module()) || (@project.project? && property.project())
+ }
+ properties.each do |property|
+ category = (property.category().present? ? property.category() : plugin.name())
+ @properties_per_category[category]||=[]
+ @properties_per_category[category]<<property
+ end
+ end
end
diff --git a/sonar-server/src/main/webapp/WEB-INF/app/controllers/settings_controller.rb b/sonar-server/src/main/webapp/WEB-INF/app/controllers/settings_controller.rb
index 854c5f746db..a936793ac29 100644
--- a/sonar-server/src/main/webapp/WEB-INF/app/controllers/settings_controller.rb
+++ b/sonar-server/src/main/webapp/WEB-INF/app/controllers/settings_controller.rb
@@ -21,25 +21,28 @@ class SettingsController < ApplicationController
SECTION=Navigation::SECTION_CONFIGURATION
- verify :method => :post, :only => ['update'], :redirect_to => { :action => :index }
+ verify :method => :post, :only => ['update'], :redirect_to => {:action => :index}
def index
- return access_denied unless is_admin?
+ return access_denied unless is_admin?
+ load_properties()
+ @category ||= 'General'
end
def update
if params[:resource_id]
project=Project.by_key(params[:resource_id])
- return access_denied unless is_admin?(project)
+ return access_denied unless (project && is_admin?(project))
resource_id=project.id
else
return access_denied unless is_admin?
+ resource_id=nil
end
- plugins = java_facade.getPluginsMetadata()
- plugins.each do |plugin|
- properties=java_facade.getPluginProperties(plugin)
- properties.each do |property|
+ load_properties()
+
+ if @category && @properties_per_category[@category]
+ @properties_per_category[@category].each do |property|
value=params[property.key()]
persisted_property = Property.find(:first, :conditions => {:prop_key=> property.key(), :resource_id => resource_id, :user_id => nil})
@@ -48,20 +51,34 @@ class SettingsController < ApplicationController
Property.delete_all('prop_key' => property.key(), 'resource_id' => resource_id, 'user_id' => nil)
elsif persisted_property.text_value != value.to_s
persisted_property.text_value = value.to_s
- persisted_property.save
+ persisted_property.save!
end
- elsif !value.blank?
+ elsif !value.blank?
Property.create(:prop_key => property.key(), :text_value => value.to_s, :resource_id => resource_id)
end
end
+ java_facade.reloadConfiguration()
+ flash[:notice] = 'Parameters updated'
end
- java_facade.reloadConfiguration()
- flash[:notice] = 'Parameters updated.'
if resource_id
- redirect_to :controller => 'project', :action => 'settings', :id => resource_id
+ redirect_to :controller => 'project', :action => 'settings', :id => resource_id, :category => @category
else
- redirect_to :action => 'index'
+ redirect_to :controller => 'settings', :action => 'index', :category => @category
+ end
+ end
+
+ private
+
+ def load_properties
+ @category=params[:category]
+ @properties_per_category={}
+ java_facade.getPluginsMetadata().each do |plugin|
+ java_facade.getPluginProperties(plugin).select { |property| property.global }.each do |property|
+ category = (property.category().present? ? property.category() : plugin.name())
+ @properties_per_category[category]||=[]
+ @properties_per_category[category]<<property
+ end
end
end
end
diff --git a/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_delete_project.html.erb b/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_delete_project.html.erb
index 7e89d570273..282a23c7627 100644
--- a/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_delete_project.html.erb
+++ b/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_delete_project.html.erb
@@ -1,5 +1,5 @@
<% if @snapshot.root? %>
-<h2>Delete project</h2>
+<h1>Delete project</h1>
<div class="yui-g widget" id="widget_delete_project">
<div class="warning">
This operation can not be undone.
diff --git a/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_exclusions.html.erb b/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_exclusions.html.erb
index 2e315bc8248..b2208db0f2b 100644
--- a/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_exclusions.html.erb
+++ b/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_exclusions.html.erb
@@ -1,4 +1,4 @@
-<h2>Exclude sources from code analysis</h2>
+<h1>Exclude sources from code analysis</h1>
<div class="yui-g widget" id="widget_exclusions">
diff --git a/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_links.html.erb b/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_links.html.erb
index 4ae38c09f8d..8f79d83807e 100644
--- a/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_links.html.erb
+++ b/sonar-server/src/main/webapp/WEB-INF/app/views/project/settings/_links.html.erb
@@ -1,4 +1,4 @@
-<h2>Project links</h2>
+<h1>Project links</h1>
<div class="yui-g widget" id="widget_links">
<% form_for( 'set_links', :url => { :action => 'set_links', :project_id => @project.id } ) do |form|
diff --git a/sonar-server/src/main/webapp/WEB-INF/app/views/settings/_plugins.html.erb b/sonar-server/src/main/webapp/WEB-INF/app/views/settings/_plugins.html.erb
index d0d7ad83c44..59e305d9573 100644
--- a/sonar-server/src/main/webapp/WEB-INF/app/views/settings/_plugins.html.erb
+++ b/sonar-server/src/main/webapp/WEB-INF/app/views/settings/_plugins.html.erb
@@ -1,130 +1,125 @@
<style type="text/css">
-#plugins .plugin {
+ #plugins .plugin {
padding: 5px;
border: 1px solid #ddd;
background-color: #fff;
- }
-#plugins .plugin h2 {
- margin-left: 10px;
- font-size: 122%;
- color: #333;
-}
-#plugins .plugin h3 {
- margin-left: 5px;
-}
-#plugins .plugin p {
- padding: 5px 5px;
- }
-#plugins .plugin img {
- padding: 5px 0 0 5px;
- }
-</style>
-<div id="plugins">
- <% form_tag :controller => :settings, :action => :update do
- plugins = controller.java_facade.getPluginsMetadata().sort {|p1,p2| p1.getName() <=> p2.getName()}
- plugin_properties={}
- plugins.each do |plugin|
- properties = controller.java_facade.getPluginProperties(plugin).select do |property|
- (project && project.module? && property.module()) or (project && project.project? && property.project()) or (project.nil? && property.global)
- end
- plugin_properties[plugin]=properties
- end
-
- %>
- <% if project %>
- <input type="hidden" name="resource_id" value="<%= project.id -%>" ></input>
- <% end %>
- <table width="100%">
- <tr>
- <td width="1%" nowrap class="column first">
- <table class="data selector">
- <thead><tr><th>
- <span>Select plugin</span>
- </th></tr></thead>
- <tbody>
- <% plugins.each do |plugin|
- if plugin_properties[plugin].empty?
- # we display only plugins with properties
- next
- end
- %>
- <tr class="select <%= cycle('even', 'odd', :name => 'plugins') -%>" id="select_<%= plugin.getKey() -%>">
- <td><a href="#" onclick="showPlugin('<%= plugin.getKey() -%>')"><%= h(plugin.getName()) -%></a></td>
- </tr>
- <% end %>
- </tbody>
- </table>
- <br/>
- <%= submit_tag('Save parameters') %>
- </td>
- <td class="column">
-
- <%
- values = Property.hash(project ? project.id : nil)
- plugins.each do |plugin|
- properties = plugin_properties[plugin]
- %>
- <div class="plugin" id="plugin_<%= plugin.getKey() -%>" style="display: none;">
- <table class="spaced">
- <% unless plugin.getDescription().blank? %>
- <tr><td class="odd">
- <p><%= plugin.getDescription() %></p>
- </td></tr>
- <% end %>
+ }
+ #plugins .plugin h2 {
+ margin-left: 10px;
+ font-size: 122%;
+ color: #333;
+ }
- <% properties.each do |property|
- value= values[property.key()] || ''
- css = cycle('even', 'odd', :name => plugin.getKey())
- %>
- <tr class="<%= css -%>">
- <td>
- <h3><%= property.name() %> <% if property.project() %><span class="note">[<%= property.key() -%>]</span><% end %></h3>
- <p><%= property.description() %></p>
- <p>
- <% span_id = "text_" + property.key().gsub('.', '_') %>
- <% textfield = text_field_tag property.key(), value, :size => '20' %>
- <% textfield += link_to_function(image_tag("zoom.png"), "replaceTextField('#{span_id}', '#{property.key()}')", :id => "toggle_text", :class => 'nolink') %>
- <% textarea = text_area_tag property.key(), value, :size => "100x10" %>
- <span id="<%= span_id %>"><%= (value.length < 50) ? textfield : textarea %></span>
+ #plugins .plugin h3 {
+ margin-left: 5px;
+ }
- <% unless property.defaultValue().blank? %>
- <% if project.nil? %>
- <span class="note">Default : <%= h property.defaultValue() -%></span>
- <% else %>
- <span class="note">Default : <%= h Property.value(property.key(), nil, property.defaultValue()) -%></span>
- <% end %>
- <% end %>
- </p>
- </td>
- </tr>
-
- <% end %>
- </table>
- </div>
- <% end %>
- </td>
- </tr>
- </table>
- <% end %>
-</div>
+ #plugins .plugin p {
+ padding: 5px 5px;
+ }
+ #plugins .plugin img {
+ padding: 5px 0 0 5px;
+ }
+</style>
<script type="text/javascript">
function replaceTextField(span_id, key) {
var text_field_value = $F(key);
var text_area = '<textarea cols="100" id="' + key + '" name="' + key + '" rows="10">' + text_field_value + '</textarea>';
$(span_id).replace(text_area);
}
- function showPlugin(id) {
- $$('.plugin').each(function(element) {
- element.hide();
- });
- $$('.select').each(function(element) {
- element.removeClassName('selected');
- });
- $('plugin_' + id).show();
- $('select_' + id).addClassName('selected');
- return false;
- }
- showPlugin('core');
-</script> \ No newline at end of file
+</script>
+<div id="plugins">
+ <table width="100%">
+ <tr>
+ <td width="1%" nowrap class="column first">
+ <table class="data selector">
+ <thead>
+ <tr>
+ <th>
+ <span>Category</span>
+ </th>
+ </tr>
+ </thead>
+ <tbody>
+ <%
+ @properties_per_category.keys.sort.each do |category|
+ unless @properties_per_category[category].empty?
+ %>
+ <tr class="select <%= cycle('even', 'odd', :name => 'category') -%> <%= 'selected' if @category==category -%>" id="select_<%= category -%>">
+ <td><%= link_to h(category), :overwrite_params => {:category => category} -%></td>
+ </tr>
+ <% end
+ end
+ %>
+ </tbody>
+ </table>
+ <br/>
+ </td>
+
+ <td class="column">
+ <% if @category
+ category_name = message("settings.category.#{@category}", :default => @category)
+ %>
+
+ <% form_tag :controller => :settings, :action => :update do %>
+ <%= hidden_field_tag('category', @category) -%>
+ <% if @project %>
+ <input type="hidden" name="resource_id" value="<%= @project.id -%>"/>
+ <% end %>
+ <table class="data marginbottom10">
+ <thead>
+ <tr>
+ <th>
+ <span><%= h(category_name) -%></span>
+ </th>
+ </tr>
+ </thead>
+ <tbody>
+ <%
+ if @properties_per_category[@category]
+ @properties_per_category[@category].each do |property|
+ value = Property.value(property.key(), (@project ? @project.id : nil), '')
+ %>
+ <tr class="<%= cycle('even', 'odd', :name => 'properties') -%>">
+ <td style="padding: 10px">
+ <h3>
+ <%= h(property.name()) -%>
+ <% if property.project() %>
+ <br/><span class="note"><%= property.key() -%></span>
+ <% end %>
+ </h3>
+ <% if property.description.present? %>
+ <p class="marginbottom10"><%= property.description -%></p>
+ <% end %>
+ <p>
+ <% span_id = "text_" + property.key().gsub('.', '_') %>
+ <% textfield = text_field_tag property.key(), value, :size => '20' %>
+ <% textfield += link_to_function(image_tag("zoom.png"), "replaceTextField('#{span_id}', '#{property.key()}')", :id => "toggle_text", :class => 'nolink') %>
+ <% textarea = text_area_tag property.key(), value, :size => "100x10" %>
+ <span id="<%= span_id %>"><%= (value.length < 50) ? textfield : textarea %></span>
+
+ <% unless property.defaultValue().blank? %>
+ <% if @project %>
+ <span class="note">Default : <%= h Property.value(property.key(), nil, property.defaultValue()) -%></span>
+ <% else %>
+ <span class="note">Default : <%= h property.defaultValue() -%></span>
+ <% end %>
+ <% end %>
+ </p>
+ </td>
+ </tr>
+ <% end
+ end
+ %>
+ </tbody>
+ </table>
+ <% save_message=message('settings.save_category', :params => [category_name]) %>
+ <%= submit_tag(save_message, :disable_with => save_message, :id => 'save') -%>
+ <% end %>
+ <% end %>
+ </td>
+ </tr>
+ </table>
+</div>