Commit 117cdd09dde08e2fd6df291330493ec4e17b6494
Exists in
master
and in
23 other branches
Merge branch 'master' into tolerance-time
Conflicts: app/helpers/application_helper.rb app/views/content_viewer/_article_toolbar.rhtml test/functional/content_viewer_controller_test.rb
Showing
289 changed files
with
12957 additions
and
8050 deletions
 
Show diff stats
Too many changes.
To preserve performance only 100 of 289 files displayed.
AUTHORS
| ... | ... | @@ -22,6 +22,7 @@ Diego Araújo <diegoamc90@gmail.com> | 
| 22 | 22 | Diego Araújo + João M. M. da Silva <diegoamc90@gmail.com> | 
| 23 | 23 | Fernanda Lopes <nanda.listas+psl@gmail.com> | 
| 24 | 24 | Grazieno Pellegrino <grazieno@gmail.com> | 
| 25 | +Isaac Canan <isaac@intelletto.com.br> | |
| 25 | 26 | Italo Valcy <italo@dcc.ufba.br> | 
| 26 | 27 | João da Silva <jaodsilv@linux.ime.usp.br> | 
| 27 | 28 | João M. M. da Silva + Carlos Morais <jaodsilv@linux.ime.usp.br> | ... | ... | 
| ... | ... | @@ -0,0 +1,50 @@ | 
| 1 | +class LicensesController < AdminController | |
| 2 | + protect 'manage_environment_licenses', :environment | |
| 3 | + | |
| 4 | + def index | |
| 5 | + @licenses = environment.licenses | |
| 6 | + end | |
| 7 | + | |
| 8 | + def create | |
| 9 | + @license = License.new(params[:license]) | |
| 10 | + if request.post? | |
| 11 | + begin | |
| 12 | + @license.environment = environment | |
| 13 | + @license.save! | |
| 14 | + session[:notice] = _('License created') | |
| 15 | + redirect_to :action => 'index' | |
| 16 | + rescue | |
| 17 | + session[:notice] = _('License could not be created') | |
| 18 | + end | |
| 19 | + end | |
| 20 | + end | |
| 21 | + | |
| 22 | + def edit | |
| 23 | + @license = environment.licenses.find(params[:license_id]) | |
| 24 | + if request.post? | |
| 25 | + begin | |
| 26 | + @license.update_attributes!(params[:license]) | |
| 27 | + session[:notice] = _('License updated') | |
| 28 | + redirect_to :action => 'index' | |
| 29 | + rescue | |
| 30 | + session[:notice] = _('License could not be updated') | |
| 31 | + end | |
| 32 | + end | |
| 33 | + end | |
| 34 | + | |
| 35 | + def remove | |
| 36 | + @license = environment.licenses.find(params[:license_id]) | |
| 37 | + if request.post? | |
| 38 | + begin | |
| 39 | + @license.destroy | |
| 40 | + session[:notice] = _('License removed') | |
| 41 | + rescue | |
| 42 | + session[:notice] = _('License could not be removed') | |
| 43 | + end | |
| 44 | + else | |
| 45 | + session[:notice] = _('License could not be removed') | |
| 46 | + end | |
| 47 | + redirect_to :action => 'index' | |
| 48 | + end | |
| 49 | + | |
| 50 | +end | ... | ... | 
| ... | ... | @@ -0,0 +1,52 @@ | 
| 1 | +class TemplatesController < AdminController | |
| 2 | + protect 'manage_environment_templates', :environment | |
| 3 | + | |
| 4 | + def create_person_template | |
| 5 | + if request.post? | |
| 6 | + begin | |
| 7 | + identifier = params[:name].to_slug | |
| 8 | + password = Digest::MD5.hexdigest(rand.to_s) | |
| 9 | + template = User.new(:login => identifier, :email => identifier+'@templates.noo', :password => password, :password_confirmation => password, :person_data => {:name => params[:name], :is_template => true}) | |
| 10 | + template.save! | |
| 11 | + session[:notice] = _('New template created') | |
| 12 | + redirect_to :action => 'index' | |
| 13 | + rescue | |
| 14 | + @error = _('Name has already been taken') | |
| 15 | + end | |
| 16 | + end | |
| 17 | + end | |
| 18 | + | |
| 19 | + def create_community_template | |
| 20 | + if request.post? | |
| 21 | + begin | |
| 22 | + create_organization_template(Community) | |
| 23 | + session[:notice] = _('New template created') | |
| 24 | + redirect_to :action => 'index' | |
| 25 | + rescue | |
| 26 | + @error = _('Name has already been taken') | |
| 27 | + end | |
| 28 | + end | |
| 29 | + end | |
| 30 | + | |
| 31 | + def create_enterprise_template | |
| 32 | + if request.post? | |
| 33 | + begin | |
| 34 | + create_organization_template(Enterprise) | |
| 35 | + session[:notice] = _('New template created') | |
| 36 | + redirect_to :action => 'index' | |
| 37 | + rescue | |
| 38 | + @error = _('Name has already been taken') | |
| 39 | + end | |
| 40 | + end | |
| 41 | + end | |
| 42 | + | |
| 43 | + private | |
| 44 | + | |
| 45 | + def create_organization_template(klass) | |
| 46 | + identifier = params[:name].to_slug | |
| 47 | + template = klass.new(:name => params[:name], :identifier => identifier, :is_template => true) | |
| 48 | + template.save! | |
| 49 | + end | |
| 50 | + | |
| 51 | +end | |
| 52 | + | ... | ... | 
app/controllers/public/content_viewer_controller.rb
| ... | ... | @@ -101,6 +101,14 @@ class ContentViewerController < ApplicationController | 
| 101 | 101 | @images = @images.paginate(:per_page => per_page, :page => params[:npage]) unless params[:slideshow] | 
| 102 | 102 | end | 
| 103 | 103 | |
| 104 | + @unfollow_form = params[:unfollow] && params[:unfollow] == 'true' | |
| 105 | + if params[:unfollow] && params[:unfollow] == 'commit' && request.post? | |
| 106 | + @page.followers -= [params[:email]] | |
| 107 | + if @page.save | |
| 108 | + session[:notice] = _("Notification of new comments to '%s' was successfully canceled") % params[:email] | |
| 109 | + end | |
| 110 | + end | |
| 111 | + | |
| 104 | 112 | @comments = @page.comments(true).as_thread | 
| 105 | 113 | @comments_count = @page.comments.count | 
| 106 | 114 | if params[:slideshow] | ... | ... | 
app/controllers/public/search_controller.rb
| ... | ... | @@ -243,7 +243,7 @@ class SearchController < PublicController | 
| 243 | 243 | @searching = {} | 
| 244 | 244 | @titles = {} | 
| 245 | 245 | @enabled_searches.each do |key, name| | 
| 246 | - @titles[key] = name | |
| 246 | + @titles[key] = _(name) | |
| 247 | 247 | @searching[key] = params[:action] == 'index' || params[:action] == key.to_s | 
| 248 | 248 | end | 
| 249 | 249 | @names = @titles if @names.nil? | ... | ... | 
app/helpers/application_helper.rb
| ... | ... | @@ -10,6 +10,8 @@ module ApplicationHelper | 
| 10 | 10 | |
| 11 | 11 | include ThickboxHelper | 
| 12 | 12 | |
| 13 | + include ColorboxHelper | |
| 14 | + | |
| 13 | 15 | include BoxesHelper | 
| 14 | 16 | |
| 15 | 17 | include FormsHelper | 
| ... | ... | @@ -1087,24 +1089,24 @@ module ApplicationHelper | 
| 1087 | 1089 | |
| 1088 | 1090 | def search_contents_menu | 
| 1089 | 1091 | links = [ | 
| 1090 | - {s_('contents|More Recent') => {:href => url_for({:controller => 'search', :action => 'contents', :filter => 'more_recent'})}}, | |
| 1091 | - {s_('contents|More Viewed') => {:href => url_for({:controller => 'search', :action => 'contents', :filter => 'more_popular'})}}, | |
| 1092 | - {s_('contents|Most Commented') => {:href => url_for({:controller => 'search', :action => 'contents', :filter => 'more_comments'})}} | |
| 1092 | + {s_('contents|More recent') => {:href => url_for({:controller => 'search', :action => 'contents', :filter => 'more_recent'})}}, | |
| 1093 | + {s_('contents|More viewed') => {:href => url_for({:controller => 'search', :action => 'contents', :filter => 'more_popular'})}}, | |
| 1094 | + {s_('contents|Most commented') => {:href => url_for({:controller => 'search', :action => 'contents', :filter => 'more_comments'})}} | |
| 1093 | 1095 | ] | 
| 1094 | 1096 | if logged_in? | 
| 1095 | - links.push(_('New Content') => lightbox_options({:href => url_for({:controller => 'cms', :action => 'new', :profile => current_user.login, :cms => true})})) | |
| 1097 | + links.push(_('New content') => colorbox_options({:href => url_for({:controller => 'cms', :action => 'new', :profile => current_user.login, :cms => true})})) | |
| 1096 | 1098 | end | 
| 1097 | 1099 | |
| 1098 | 1100 | link_to(content_tag(:span, _('Contents'), :class => 'icon-menu-articles'), {:controller => "search", :action => 'contents', :category_path => ''}, :id => 'submenu-contents') + | 
| 1099 | - link_to(content_tag(:span, _('Contents Menu')), '#', :onclick => "toggleSubmenu(this,'',#{links.to_json}); return false", :class => 'menu-submenu-trigger up', :id => 'submenu-contents-trigger') | |
| 1101 | + link_to(content_tag(:span, _('Contents menu')), '#', :onclick => "toggleSubmenu(this,'',#{links.to_json}); return false", :class => 'menu-submenu-trigger up', :id => 'submenu-contents-trigger') | |
| 1100 | 1102 | end | 
| 1101 | 1103 | alias :browse_contents_menu :search_contents_menu | 
| 1102 | 1104 | |
| 1103 | 1105 | def search_people_menu | 
| 1104 | 1106 | links = [ | 
| 1105 | - {s_('people|More Recent') => {:href => url_for({:controller => 'search', :action => 'people', :filter => 'more_recent'})}}, | |
| 1106 | - {s_('people|More Active') => {:href => url_for({:controller => 'search', :action => 'people', :filter => 'more_active'})}}, | |
| 1107 | - {s_('people|More Popular') => {:href => url_for({:controller => 'search', :action => 'people', :filter => 'more_popular'})}} | |
| 1107 | + {s_('people|More recent') => {:href => url_for({:controller => 'search', :action => 'people', :filter => 'more_recent'})}}, | |
| 1108 | + {s_('people|More active') => {:href => url_for({:controller => 'search', :action => 'people', :filter => 'more_active'})}}, | |
| 1109 | + {s_('people|More popular') => {:href => url_for({:controller => 'search', :action => 'people', :filter => 'more_popular'})}} | |
| 1108 | 1110 | ] | 
| 1109 | 1111 | if logged_in? | 
| 1110 | 1112 | links.push(_('My friends') => {:href => url_for({:profile => current_user.login, :controller => 'friends'})}) | 
| ... | ... | @@ -1112,15 +1114,15 @@ module ApplicationHelper | 
| 1112 | 1114 | end | 
| 1113 | 1115 | |
| 1114 | 1116 | link_to(content_tag(:span, _('People'), :class => 'icon-menu-people'), {:controller => "search", :action => 'people', :category_path => ''}, :id => 'submenu-people') + | 
| 1115 | - link_to(content_tag(:span, _('People Menu')), '#', :onclick => "toggleSubmenu(this,'',#{links.to_json}); return false", :class => 'menu-submenu-trigger up', :id => 'submenu-people-trigger') | |
| 1117 | + link_to(content_tag(:span, _('People menu')), '#', :onclick => "toggleSubmenu(this,'',#{links.to_json}); return false", :class => 'menu-submenu-trigger up', :id => 'submenu-people-trigger') | |
| 1116 | 1118 | end | 
| 1117 | 1119 | alias :browse_people_menu :search_people_menu | 
| 1118 | 1120 | |
| 1119 | 1121 | def search_communities_menu | 
| 1120 | 1122 | links = [ | 
| 1121 | - {s_('communities|More Recent') => {:href => url_for({:controller => 'search', :action => 'communities', :filter => 'more_recent'})}}, | |
| 1122 | - {s_('communities|More Active') => {:href => url_for({:controller => 'search', :action => 'communities', :filter => 'more_active'})}}, | |
| 1123 | - {s_('communities|More Popular') => {:href => url_for({:controller => 'search', :action => 'communities', :filter => 'more_popular'})}} | |
| 1123 | + {s_('communities|More recent') => {:href => url_for({:controller => 'search', :action => 'communities', :filter => 'more_recent'})}}, | |
| 1124 | + {s_('communities|More active') => {:href => url_for({:controller => 'search', :action => 'communities', :filter => 'more_active'})}}, | |
| 1125 | + {s_('communities|More popular') => {:href => url_for({:controller => 'search', :action => 'communities', :filter => 'more_popular'})}} | |
| 1124 | 1126 | ] | 
| 1125 | 1127 | if logged_in? | 
| 1126 | 1128 | links.push(_('My communities') => {:href => url_for({:profile => current_user.login, :controller => 'memberships'})}) | 
| ... | ... | @@ -1128,7 +1130,7 @@ module ApplicationHelper | 
| 1128 | 1130 | end | 
| 1129 | 1131 | |
| 1130 | 1132 | link_to(content_tag(:span, _('Communities'), :class => 'icon-menu-community'), {:controller => "search", :action => 'communities'}, :id => 'submenu-communities') + | 
| 1131 | - link_to(content_tag(:span, _('Communities Menu')), '#', :onclick => "toggleSubmenu(this,'',#{links.to_json}); return false", :class => 'menu-submenu-trigger up', :id => 'submenu-communities-trigger') | |
| 1133 | + link_to(content_tag(:span, _('Communities menu')), '#', :onclick => "toggleSubmenu(this,'',#{links.to_json}); return false", :class => 'menu-submenu-trigger up', :id => 'submenu-communities-trigger') | |
| 1132 | 1134 | end | 
| 1133 | 1135 | alias :browse_communities_menu :search_communities_menu | 
| 1134 | 1136 | |
| ... | ... | @@ -1181,6 +1183,10 @@ module ApplicationHelper | 
| 1181 | 1183 | ], :class => 'limited-text-area') | 
| 1182 | 1184 | end | 
| 1183 | 1185 | |
| 1186 | + def expandable_text_area(object_name, method, text_area_id, options = {}) | |
| 1187 | + text_area(object_name, method, { :id => text_area_id, :onkeyup => "grow_text_area('#{text_area_id}')" }.merge(options)) | |
| 1188 | + end | |
| 1189 | + | |
| 1184 | 1190 | def pluralize_without_count(count, singular, plural = nil) | 
| 1185 | 1191 | count == 1 ? singular : (plural || singular.pluralize) | 
| 1186 | 1192 | end | 
| ... | ... | @@ -1342,4 +1348,21 @@ module ApplicationHelper | 
| 1342 | 1348 | def remove_content_button(action) | 
| 1343 | 1349 | @plugins.dispatch("content_remove_#{action.to_s}", @page).include?(true) | 
| 1344 | 1350 | end | 
| 1351 | + | |
| 1352 | + def template_options(klass, field_name) | |
| 1353 | + return '' if klass.templates.count == 0 | |
| 1354 | + return hidden_field_tag("#{field_name}[template_id]", klass.templates.first.id) if klass.templates.count == 1 | |
| 1355 | + | |
| 1356 | + counter = 0 | |
| 1357 | + radios = klass.templates.map do |template| | |
| 1358 | + counter += 1 | |
| 1359 | + content_tag('li', labelled_radio_button(template.name, "#{field_name}[template_id]", template.id, counter==1)) | |
| 1360 | + end.join("\n") | |
| 1361 | + | |
| 1362 | + content_tag('div', content_tag('span', _('Template:')) + | |
| 1363 | + content_tag('ul', radios, :style => 'list-style: none; padding-left: 0; margin-top: 0.5em;'), | |
| 1364 | + :id => 'template-options', | |
| 1365 | + :style => 'margin-top: 1em' | |
| 1366 | + ) | |
| 1367 | + end | |
| 1345 | 1368 | end | ... | ... | 
| ... | ... | @@ -0,0 +1,21 @@ | 
| 1 | +module ColorboxHelper | |
| 2 | + | |
| 3 | + def colorbox_close_button(text, options = {}) | |
| 4 | + button(:close, text, '#', colorbox_options(options, :close)) | |
| 5 | + end | |
| 6 | + | |
| 7 | + def colorbox_button(type, label, url, options = {}) | |
| 8 | + button(type, label, url, colorbox_options(options)) | |
| 9 | + end | |
| 10 | + | |
| 11 | + # options must be an HTML options hash as passed to link_to etc. | |
| 12 | + # | |
| 13 | + # returns a new hash with colorbox class added. Keeps existing classes. | |
| 14 | + def colorbox_options(options, type=nil) | |
| 15 | + the_class = 'colorbox' | |
| 16 | + the_class += "-#{type.to_s}" unless type.nil? | |
| 17 | + the_class << " #{options[:class]}" if options.has_key?(:class) | |
| 18 | + options.merge(:class => the_class) | |
| 19 | + end | |
| 20 | + | |
| 21 | +end | ... | ... | 
app/helpers/search_helper.rb
| ... | ... | @@ -18,7 +18,7 @@ module SearchHelper | 
| 18 | 18 | |
| 19 | 19 | SortOptions = { | 
| 20 | 20 | :products => ActiveSupport::OrderedHash[ :none, {:label => _('Relevance')}, | 
| 21 | - :more_recent, {:label => _('More Recent'), :solr_opts => {:sort => 'updated_at desc, score desc'}}, | |
| 21 | + :more_recent, {:label => _('More recent'), :solr_opts => {:sort => 'updated_at desc, score desc'}}, | |
| 22 | 22 | :name, {:label => _('Name'), :solr_opts => {:sort => 'name_sortable asc'}}, | 
| 23 | 23 | :closest, {:label => _('Closest to me'), :if => proc{ logged_in? && (profile=current_user.person).lat && profile.lng }, | 
| 24 | 24 | :solr_opts => {:sort => "geodist() asc", | ... | ... | 
app/models/article.rb
| ... | ... | @@ -34,9 +34,12 @@ class Article < ActiveRecord::Base | 
| 34 | 34 | settings_items :display_hits, :type => :boolean, :default => true | 
| 35 | 35 | settings_items :author_name, :type => :string, :default => "" | 
| 36 | 36 | settings_items :allow_members_to_edit, :type => :boolean, :default => false | 
| 37 | + settings_items :followers, :type => Array, :default => [] | |
| 37 | 38 | |
| 38 | 39 | belongs_to :reference_article, :class_name => "Article", :foreign_key => 'reference_article_id' | 
| 39 | 40 | |
| 41 | + belongs_to :license | |
| 42 | + | |
| 40 | 43 | has_many :translations, :class_name => 'Article', :foreign_key => :translation_of_id | 
| 41 | 44 | belongs_to :translation_of, :class_name => 'Article', :foreign_key => :translation_of_id | 
| 42 | 45 | before_destroy :rotate_translations | 
| ... | ... | @@ -51,6 +54,11 @@ class Article < ActiveRecord::Base | 
| 51 | 54 | end | 
| 52 | 55 | end | 
| 53 | 56 | |
| 57 | + after_destroy :destroy_activity | |
| 58 | + def destroy_activity | |
| 59 | + self.activity.destroy if self.activity | |
| 60 | + end | |
| 61 | + | |
| 54 | 62 | xss_terminate :only => [ :name ], :on => 'validation', :with => 'white_list' | 
| 55 | 63 | |
| 56 | 64 | named_scope :in_category, lambda { |category| | ... | ... | 
app/models/comment.rb
| ... | ... | @@ -77,11 +77,30 @@ class Comment < ActiveRecord::Base | 
| 77 | 77 | article.comments_updated if article.kind_of?(Article) | 
| 78 | 78 | end | 
| 79 | 79 | |
| 80 | - after_create do |comment| | |
| 81 | - if comment.source.kind_of?(Article) && comment.article.notify_comments? && !comment.article.profile.notification_emails.empty? | |
| 82 | - Comment::Notifier.deliver_mail(comment) | |
| 80 | + after_create :new_follower | |
| 81 | + def new_follower | |
| 82 | + if source.kind_of?(Article) | |
| 83 | + article.followers += [author_email] | |
| 84 | + article.followers -= article.profile.notification_emails | |
| 85 | + article.followers.uniq! | |
| 86 | + article.save | |
| 87 | + end | |
| 88 | + end | |
| 89 | + | |
| 90 | + after_create :notify_by_mail | |
| 91 | + def notify_by_mail | |
| 92 | + if source.kind_of?(Article) && article.notify_comments? | |
| 93 | + if !article.profile.notification_emails.empty? | |
| 94 | + Comment::Notifier.deliver_mail(self) | |
| 95 | + end | |
| 96 | + emails = article.followers - [author_email] | |
| 97 | + if !emails.empty? | |
| 98 | + Comment::Notifier.deliver_mail_to_followers(self, emails) | |
| 99 | + end | |
| 83 | 100 | end | 
| 101 | + end | |
| 84 | 102 | |
| 103 | + after_create do |comment| | |
| 85 | 104 | if comment.source.kind_of?(Article) | 
| 86 | 105 | comment.article.create_activity if comment.article.activity.nil? | 
| 87 | 106 | if comment.article.activity | 
| ... | ... | @@ -140,6 +159,22 @@ class Comment < ActiveRecord::Base | 
| 140 | 159 | :environment => profile.environment.name, | 
| 141 | 160 | :url => profile.environment.top_url | 
| 142 | 161 | end | 
| 162 | + def mail_to_followers(comment, emails) | |
| 163 | + profile = comment.article.profile | |
| 164 | + bcc emails | |
| 165 | + from "#{profile.environment.name} <#{profile.environment.contact_email}>" | |
| 166 | + subject _("[%s] %s commented on a content of %s") % [profile.environment.name, comment.author_name, profile.short_name] | |
| 167 | + body :recipient => profile.nickname || profile.name, | |
| 168 | + :sender => comment.author_name, | |
| 169 | + :sender_link => comment.author_link, | |
| 170 | + :article_title => comment.article.name, | |
| 171 | + :comment_url => comment.url, | |
| 172 | + :unsubscribe_url => comment.article.view_url.merge({:unfollow => true}), | |
| 173 | + :comment_title => comment.title, | |
| 174 | + :comment_body => comment.body, | |
| 175 | + :environment => profile.environment.name, | |
| 176 | + :url => profile.environment.top_url | |
| 177 | + end | |
| 143 | 178 | end | 
| 144 | 179 | |
| 145 | 180 | def rejected? | ... | ... | 
app/models/community.rb
| ... | ... | @@ -61,10 +61,10 @@ class Community < Organization | 
| 61 | 61 | |
| 62 | 62 | def name=(value) | 
| 63 | 63 | super(value) | 
| 64 | - self.identifier = value.to_slug | |
| 64 | + self.identifier ||= value.to_slug | |
| 65 | 65 | end | 
| 66 | 66 | |
| 67 | - def template | |
| 67 | + def default_template | |
| 68 | 68 | environment.community_template | 
| 69 | 69 | end | 
| 70 | 70 | ... | ... | 
app/models/enterprise.rb
| ... | ... | @@ -154,13 +154,14 @@ class Enterprise < Organization | 
| 154 | 154 | true | 
| 155 | 155 | end | 
| 156 | 156 | |
| 157 | - def template | |
| 158 | - if enabled? | |
| 159 | - environment.enterprise_template | |
| 160 | - else | |
| 161 | - environment.inactive_enterprise_template | |
| 162 | - end | |
| 157 | + def default_template | |
| 158 | + environment.enterprise_template | |
| 159 | + end | |
| 160 | + | |
| 161 | + def template_with_inactive_enterprise | |
| 162 | + !enabled? ? environment.inactive_enterprise_template : template_without_inactive_enterprise | |
| 163 | 163 | end | 
| 164 | + alias_method_chain :template, :inactive_enterprise | |
| 164 | 165 | |
| 165 | 166 | def control_panel_settings_button | 
| 166 | 167 | {:title => __('Enterprise Info and settings'), :icon => 'edit-profile-enterprise'} | ... | ... | 
app/models/environment.rb
| ... | ... | @@ -24,6 +24,8 @@ class Environment < ActiveRecord::Base | 
| 24 | 24 | 'manage_environment_roles' => N_('Manage environment roles'), | 
| 25 | 25 | 'manage_environment_validators' => N_('Manage environment validators'), | 
| 26 | 26 | 'manage_environment_users' => N_('Manage environment users'), | 
| 27 | + 'manage_environment_templates' => N_('Manage environment templates'), | |
| 28 | + 'manage_environment_licenses' => N_('Manage environment licenses'), | |
| 27 | 29 | } | 
| 28 | 30 | |
| 29 | 31 | module Roles | 
| ... | ... | @@ -158,6 +160,7 @@ class Environment < ActiveRecord::Base | 
| 158 | 160 | has_many :products, :through => :enterprises | 
| 159 | 161 | has_many :people | 
| 160 | 162 | has_many :communities | 
| 163 | + has_many :licenses | |
| 161 | 164 | |
| 162 | 165 | has_many :categories | 
| 163 | 166 | has_many :display_categories, :class_name => 'Category', :conditions => 'display_color is not null and parent_id is null', :order => 'display_color' | 
| ... | ... | @@ -719,12 +722,12 @@ class Environment < ActiveRecord::Base | 
| 719 | 722 | |
| 720 | 723 | def create_templates | 
| 721 | 724 | pre = self.name.to_slug + '_' | 
| 722 | - ent_id = Enterprise.create!(:name => 'Enterprise template', :identifier => pre + 'enterprise_template', :environment => self, :visible => false).id | |
| 723 | - inactive_enterprise_tmpl = Enterprise.create!(:name => 'Inactive Enterprise template', :identifier => pre + 'inactive_enterprise_template', :environment => self, :visible => false) | |
| 724 | - com_id = Community.create!(:name => 'Community template', :identifier => pre + 'community_template', :environment => self, :visible => false).id | |
| 725 | + ent_id = Enterprise.create!(:name => 'Enterprise template', :identifier => pre + 'enterprise_template', :environment => self, :visible => false, :is_template => true).id | |
| 726 | + inactive_enterprise_tmpl = Enterprise.create!(:name => 'Inactive Enterprise template', :identifier => pre + 'inactive_enterprise_template', :environment => self, :visible => false, :is_template => true) | |
| 727 | + com_id = Community.create!(:name => 'Community template', :identifier => pre + 'community_template', :environment => self, :visible => false, :is_template => true).id | |
| 725 | 728 | pass = Digest::MD5.hexdigest rand.to_s | 
| 726 | 729 | user = User.create!(:login => (pre + 'person_template'), :email => (pre + 'template@template.noo'), :password => pass, :password_confirmation => pass, :environment => self).person | 
| 727 | - user.update_attributes(:visible => false, :name => "Person template") | |
| 730 | + user.update_attributes(:visible => false, :name => "Person template", :is_template => true) | |
| 728 | 731 | usr_id = user.id | 
| 729 | 732 | self.settings[:enterprise_template_id] = ent_id | 
| 730 | 733 | self.inactive_enterprise_template = inactive_enterprise_tmpl | 
| ... | ... | @@ -740,6 +743,18 @@ class Environment < ActiveRecord::Base | 
| 740 | 743 | end | 
| 741 | 744 | end | 
| 742 | 745 | |
| 746 | + after_create :create_default_licenses | |
| 747 | + def create_default_licenses | |
| 748 | + License.create!(:name => 'CC (by)', :url => 'http://creativecommons.org/licenses/by/3.0/legalcode', :environment => self) | |
| 749 | + License.create!(:name => 'CC (by-nd)', :url => 'http://creativecommons.org/licenses/by-nd/3.0/legalcode', :environment => self) | |
| 750 | + License.create!(:name => 'CC (by-sa)', :url => 'http://creativecommons.org/licenses/by-sa/3.0/legalcode', :environment => self) | |
| 751 | + License.create!(:name => 'CC (by-nc)', :url => 'http://creativecommons.org/licenses/by-nc/3.0/legalcode', :environment => self) | |
| 752 | + License.create!(:name => 'CC (by-nc-nd)', :url => 'http://creativecommons.org/licenses/by-nc-nd/3.0/legalcode', :environment => self) | |
| 753 | + License.create!(:name => 'CC (by-nc-sa)', :url => 'http://creativecommons.org/licenses/by-nc-sa/3.0/legalcode', :environment => self) | |
| 754 | + License.create!(:name => 'Free Art', :url => 'http://artlibre.org/licence/lal/en', :environment => self) | |
| 755 | + License.create!(:name => 'GNU FDL', :url => 'http://www.gnu.org/licenses/fdl-1.3.txt', :environment => self) | |
| 756 | + end | |
| 757 | + | |
| 743 | 758 | def highlighted_products_with_image(options = {}) | 
| 744 | 759 | Product.find(:all, {:conditions => {:highlighted => true, :enterprise_id => self.enterprises.find(:all, :select => :id) }, :joins => :image}.merge(options)) | 
| 745 | 760 | end | ... | ... | 
app/models/input.rb
| ... | ... | @@ -0,0 +1,12 @@ | 
| 1 | +class License < ActiveRecord::Base | |
| 2 | + belongs_to :environment | |
| 3 | + has_many :content, :class_name => 'Article', :foreign_key => 'license_id' | |
| 4 | + | |
| 5 | + validates_presence_of :name, :environment | |
| 6 | + validates_presence_of :slug, :if => lambda {|license| license.name.present?} | |
| 7 | + validates_uniqueness_of :slug, :scope => :environment_id | |
| 8 | + | |
| 9 | + before_validation do |license| | |
| 10 | + license.slug ||= license.name.to_slug if license.name.present? | |
| 11 | + end | |
| 12 | +end | ... | ... | 
app/models/location_block.rb
| 1 | 1 | class LocationBlock < Block | 
| 2 | 2 | |
| 3 | - settings_items :zoom, :type => :integer , :default => 4 | |
| 4 | - settings_items :map_type, :type => :string , :default => 'roadmap' | |
| 3 | + settings_items :zoom, :type => :integer, :default => 4 | |
| 4 | + settings_items :map_type, :type => :string, :default => 'roadmap' | |
| 5 | 5 | |
| 6 | 6 | def self.description | 
| 7 | 7 | _('Location map') | 
| ... | ... | @@ -12,18 +12,10 @@ class LocationBlock < Block | 
| 12 | 12 | end | 
| 13 | 13 | |
| 14 | 14 | def content(args={}) | 
| 15 | + block = self | |
| 15 | 16 | profile = self.owner | 
| 16 | - title = self.title | |
| 17 | - if profile.lat | |
| 18 | - block_title(title) + | |
| 19 | - content_tag('div', | |
| 20 | - '<img src="http://maps.google.com/maps/api/staticmap?center=' + profile.lat.to_s() + | |
| 21 | - ',' + profile.lng.to_s() + '&zoom=' + zoom.to_s() + | |
| 22 | - '&size=190x250&maptype=' + map_type + '&markers=' + profile.lat.to_s() + ',' + | |
| 23 | - profile.lng.to_s() + ',green' + '&sensor=false"/>', | |
| 24 | - :class => 'the-localization-map' ) | |
| 25 | - else | |
| 26 | - content_tag('i', _('This profile has no geographical position registered.')) | |
| 17 | + lambda do | |
| 18 | + render :file => 'blocks/location', :locals => {:block => block, :profile => profile} | |
| 27 | 19 | end | 
| 28 | 20 | end | 
| 29 | 21 | ... | ... | 
app/models/mailing.rb
| ... | ... | @@ -9,7 +9,11 @@ class Mailing < ActiveRecord::Base | 
| 9 | 9 | xss_terminate :only => [ :subject, :body ], :with => 'white_list', :on => 'validation' | 
| 10 | 10 | |
| 11 | 11 | after_create do |mailing| | 
| 12 | - Delayed::Job.enqueue MailingJob.new(mailing.id) | |
| 12 | + mailing.schedule | |
| 13 | + end | |
| 14 | + | |
| 15 | + def schedule | |
| 16 | + Delayed::Job.enqueue MailingJob.new(self.id) | |
| 13 | 17 | end | 
| 14 | 18 | |
| 15 | 19 | def generate_from | 
| ... | ... | @@ -30,8 +34,14 @@ class Mailing < ActiveRecord::Base | 
| 30 | 34 | |
| 31 | 35 | def deliver | 
| 32 | 36 | each_recipient do |recipient| | 
| 33 | - Mailing::Sender.deliver_mail(self, recipient.email) | |
| 34 | - self.mailing_sents.create(:person => recipient) | |
| 37 | + begin | |
| 38 | + Mailing::Sender.deliver_mail(self, recipient.email) | |
| 39 | + self.mailing_sents.create(:person => recipient) | |
| 40 | + rescue Exception | |
| 41 | + # FIXME should not discard errors silently. An idea is to collect all | |
| 42 | + # errors and generate a task (notification) for the +source+ | |
| 43 | + # (environment/organization) listing these errors. | |
| 44 | + end | |
| 35 | 45 | end | 
| 36 | 46 | end | 
| 37 | 47 | ... | ... | 
app/models/organization.rb
app/models/person.rb
app/models/price_detail.rb
| ... | ... | @@ -4,9 +4,13 @@ class PriceDetail < ActiveRecord::Base | 
| 4 | 4 | validates_presence_of :product_id | 
| 5 | 5 | |
| 6 | 6 | belongs_to :production_cost | 
| 7 | - validates_presence_of :production_cost_id | |
| 7 | + validates_presence_of :production_cost | |
| 8 | 8 | validates_uniqueness_of :production_cost_id, :scope => :product_id | 
| 9 | 9 | |
| 10 | + def name | |
| 11 | + production_cost.name | |
| 12 | + end | |
| 13 | + | |
| 10 | 14 | def price | 
| 11 | 15 | self[:price] || 0 | 
| 12 | 16 | end | ... | ... | 
app/models/product.rb
| ... | ... | @@ -158,7 +158,7 @@ class Product < ActiveRecord::Base | 
| 158 | 158 | |
| 159 | 159 | def inputs_cost | 
| 160 | 160 | return 0 if inputs.empty? | 
| 161 | - inputs.map(&:cost).inject { |sum,price| sum + price } | |
| 161 | + inputs.relevant_to_price.map(&:cost).inject { |sum,price| sum + price } | |
| 162 | 162 | end | 
| 163 | 163 | |
| 164 | 164 | def total_production_cost | 
| ... | ... | @@ -201,6 +201,7 @@ class Product < ActiveRecord::Base | 
| 201 | 201 | self.inputs(true).each{ |i| t_i += 1; se_i += 1 if i.is_from_solidarity_economy } | 
| 202 | 202 | t_i = 1 if t_i == 0 # avoid division by 0 | 
| 203 | 203 | p = case (se_i.to_f/t_i)*100 | 
| 204 | + when 0 then [0, ''] | |
| 204 | 205 | when 0..24.999 then [0, _("0%")]; | 
| 205 | 206 | when 25..49.999 then [25, _("25%")]; | 
| 206 | 207 | when 50..74.999 then [50, _("50%")]; | ... | ... | 
app/models/production_cost.rb
app/models/profile.rb
| ... | ... | @@ -65,6 +65,7 @@ class Profile < ActiveRecord::Base | 
| 65 | 65 | #FIXME: these will work only if the subclass is already loaded | 
| 66 | 66 | named_scope :enterprises, lambda { {:conditions => (Enterprise.send(:subclasses).map(&:name) << 'Enterprise').map { |klass| "profiles.type = '#{klass}'"}.join(" OR ")} } | 
| 67 | 67 | named_scope :communities, lambda { {:conditions => (Community.send(:subclasses).map(&:name) << 'Community').map { |klass| "profiles.type = '#{klass}'"}.join(" OR ")} } | 
| 68 | + named_scope :templates, :conditions => {:is_template => true} | |
| 68 | 69 | |
| 69 | 70 | def members | 
| 70 | 71 | Person.members_of(self) | 
| ... | ... | @@ -98,6 +99,7 @@ class Profile < ActiveRecord::Base | 
| 98 | 99 | has_many :action_tracker_notifications, :foreign_key => 'profile_id' | 
| 99 | 100 | has_many :tracked_notifications, :through => :action_tracker_notifications, :source => :action_tracker, :order => 'updated_at DESC' | 
| 100 | 101 | has_many :scraps_received, :class_name => 'Scrap', :foreign_key => :receiver_id, :order => "updated_at DESC", :dependent => :destroy | 
| 102 | + belongs_to :template, :class_name => 'Profile', :foreign_key => 'template_id' | |
| 101 | 103 | |
| 102 | 104 | # FIXME ugly workaround | 
| 103 | 105 | def self.human_attribute_name(attrib) | 
| ... | ... | @@ -204,7 +206,7 @@ class Profile < ActiveRecord::Base | 
| 204 | 206 | end | 
| 205 | 207 | |
| 206 | 208 | belongs_to :region | 
| 207 | - | |
| 209 | + | |
| 208 | 210 | def location(separator = ' - ') | 
| 209 | 211 | myregion = self.region | 
| 210 | 212 | if myregion | 
| ... | ... | @@ -274,8 +276,14 @@ class Profile < ActiveRecord::Base | 
| 274 | 276 | validates_format_of :identifier, :with => IDENTIFIER_FORMAT, :if => lambda { |profile| !profile.identifier.blank? } | 
| 275 | 277 | validates_exclusion_of :identifier, :in => RESERVED_IDENTIFIERS | 
| 276 | 278 | validates_uniqueness_of :identifier, :scope => :environment_id | 
| 277 | - | |
| 278 | 279 | validates_length_of :nickname, :maximum => 16, :allow_nil => true | 
| 280 | + validate :valid_template | |
| 281 | + | |
| 282 | + def valid_template | |
| 283 | + if template_id.present? and !template.is_template | |
| 284 | + errors.add(:template, _('is not a template.')) | |
| 285 | + end | |
| 286 | + end | |
| 279 | 287 | |
| 280 | 288 | before_create :set_default_environment | 
| 281 | 289 | def set_default_environment | 
| ... | ... | @@ -285,12 +293,12 @@ class Profile < ActiveRecord::Base | 
| 285 | 293 | true | 
| 286 | 294 | end | 
| 287 | 295 | |
| 288 | - # registar callback for creating boxes after the object is created. | |
| 296 | + # registar callback for creating boxes after the object is created. | |
| 289 | 297 | after_create :create_default_set_of_boxes | 
| 290 | 298 | |
| 291 | 299 | # creates the initial set of boxes when the profile is created. Can be | 
| 292 | 300 | # overriden for each subclass to create a custom set of boxes for its | 
| 293 | - # instances. | |
| 301 | + # instances. | |
| 294 | 302 | def create_default_set_of_boxes | 
| 295 | 303 | if template | 
| 296 | 304 | apply_template(template, :copy_articles => false) | 
| ... | ... | @@ -322,10 +330,15 @@ class Profile < ActiveRecord::Base | 
| 322 | 330 | end | 
| 323 | 331 | |
| 324 | 332 | # this method should be overwritten to provide the correct template | 
| 325 | - def template | |
| 333 | + def default_template | |
| 326 | 334 | nil | 
| 327 | 335 | end | 
| 328 | 336 | |
| 337 | + def template_with_default | |
| 338 | + template_without_default || default_template | |
| 339 | + end | |
| 340 | + alias_method_chain :template, :default | |
| 341 | + | |
| 329 | 342 | def apply_template(template, options = {:copy_articles => true}) | 
| 330 | 343 | copy_blocks_from(template) | 
| 331 | 344 | copy_articles_from(template) if options[:copy_articles] | 
| ... | ... | @@ -405,7 +418,7 @@ class Profile < ActiveRecord::Base | 
| 405 | 418 | |
| 406 | 419 | # returns +false+ | 
| 407 | 420 | def person? | 
| 408 | - self.kind_of?(Person) | |
| 421 | + self.kind_of?(Person) | |
| 409 | 422 | end | 
| 410 | 423 | |
| 411 | 424 | def enterprise? | 
| ... | ... | @@ -513,7 +526,7 @@ private :generate_url, :url_options | 
| 513 | 526 | |
| 514 | 527 | after_create :insert_default_article_set | 
| 515 | 528 | def insert_default_article_set | 
| 516 | - if template | |
| 529 | + if template | |
| 517 | 530 | copy_articles_from template | 
| 518 | 531 | else | 
| 519 | 532 | default_set_of_articles.each do |article| | 
| ... | ... | @@ -583,7 +596,7 @@ private :generate_url, :url_options | 
| 583 | 596 | raise _("%s can't have members") % self.class.name | 
| 584 | 597 | end | 
| 585 | 598 | end | 
| 586 | - | |
| 599 | + | |
| 587 | 600 | def remove_member(person) | 
| 588 | 601 | self.disaffiliate(person, Profile::Roles.all_roles(environment.id)) | 
| 589 | 602 | end | 
| ... | ... | @@ -880,7 +893,7 @@ private :generate_url, :url_options | 
| 880 | 893 | |
| 881 | 894 | def self.f_enabled_proc(enabled) | 
| 882 | 895 | enabled = enabled == "true" ? true : false | 
| 883 | - enabled ? _('Enabled') : _('Not enabled') | |
| 896 | + enabled ? s_('facets|Enabled') : s_('facets|Not enabled') | |
| 884 | 897 | end | 
| 885 | 898 | def f_enabled | 
| 886 | 899 | self.enabled | 
| ... | ... | @@ -909,7 +922,7 @@ private :generate_url, :url_options | 
| 909 | 922 | acts_as_searchable :fields => facets_fields_for_solr + [:extra_data_for_index, | 
| 910 | 923 | # searched fields | 
| 911 | 924 | {:name => {:type => :text, :boost => 2.0}}, | 
| 912 | - {:identifier => :text}, {:address => :text}, {:nickname => :text}, | |
| 925 | + {:identifier => :text}, {:nickname => :text}, | |
| 913 | 926 | # filtered fields | 
| 914 | 927 | {:public => :boolean}, {:environment_id => :integer}, | 
| 915 | 928 | {:category_filter => :integer}, | 
| ... | ... | @@ -929,7 +942,7 @@ private :generate_url, :url_options | 
| 929 | 942 | |
| 930 | 943 | def control_panel_settings_button | 
| 931 | 944 | {:title => _('Profile Info and settings'), :icon => 'edit-profile'} | 
| 932 | - end | |
| 945 | + end | |
| 933 | 946 | |
| 934 | 947 | def followed_by?(person) | 
| 935 | 948 | person.is_member_of?(self) | ... | ... | 
app/models/uploaded_file.rb
| ... | ... | @@ -8,7 +8,7 @@ class UploadedFile < Article | 
| 8 | 8 | _('File') | 
| 9 | 9 | end | 
| 10 | 10 | |
| 11 | - track_actions :upload_image, :after_create, :keep_params => ["view_url", "thumbnail_path", "parent.url", "parent.name"], :if => Proc.new { |a| a.published? && a.image? && !a.parent.nil? && a.parent.gallery? }, :custom_target => :action_tracker_target | |
| 11 | + track_actions :upload_image, :after_create, :keep_params => ["view_url", "thumbnail_path", "parent.url", "parent.name"], :if => Proc.new { |a| a.published? && a.image? && !a.parent.nil? && a.parent.gallery? }, :custom_target => :parent | |
| 12 | 12 | |
| 13 | 13 | include ShortFilename | 
| 14 | 14 | |
| ... | ... | @@ -144,9 +144,4 @@ class UploadedFile < Article | 
| 144 | 144 | def uploaded_file? | 
| 145 | 145 | true | 
| 146 | 146 | end | 
| 147 | - | |
| 148 | - def action_tracker_target | |
| 149 | - self | |
| 150 | - end | |
| 151 | - | |
| 152 | 147 | end | ... | ... | 
app/models/user.rb
| ... | ... | @@ -21,12 +21,11 @@ class User < ActiveRecord::Base | 
| 21 | 21 | end | 
| 22 | 22 | end | 
| 23 | 23 | |
| 24 | - before_create :make_activation_code | |
| 25 | - | |
| 26 | 24 | before_create do |user| | 
| 27 | 25 | if user.environment.nil? | 
| 28 | 26 | user.environment = Environment.default | 
| 29 | 27 | end | 
| 28 | + user.send(:make_activation_code) unless user.environment.enabled?('skip_new_user_email_confirmation') | |
| 30 | 29 | end | 
| 31 | 30 | |
| 32 | 31 | after_create do |user| | 
| ... | ... | @@ -35,7 +34,7 @@ class User < ActiveRecord::Base | 
| 35 | 34 | user.person.name ||= user.login | 
| 36 | 35 | user.person.visible = false unless user.activated? | 
| 37 | 36 | user.person.save! | 
| 38 | - if user.environment && user.environment.enabled?('skip_new_user_email_confirmation') | |
| 37 | + if user.environment.enabled?('skip_new_user_email_confirmation') | |
| 39 | 38 | user.activate | 
| 40 | 39 | end | 
| 41 | 40 | end | 
| ... | ... | @@ -114,10 +113,11 @@ class User < ActiveRecord::Base | 
| 114 | 113 | |
| 115 | 114 | # Activates the user in the database. | 
| 116 | 115 | def activate | 
| 116 | + return false unless self.person | |
| 117 | 117 | self.activated_at = Time.now.utc | 
| 118 | 118 | self.activation_code = nil | 
| 119 | 119 | self.person.visible = true | 
| 120 | - self.person.save! && self.save | |
| 120 | + self.person.save! && self.save! | |
| 121 | 121 | end | 
| 122 | 122 | |
| 123 | 123 | def activated? | ... | ... | 
app/views/account/_signup_form.rhtml
| ... | ... | @@ -69,6 +69,8 @@ | 
| 69 | 69 | |
| 70 | 70 | <div id="signup-form-profile"> | 
| 71 | 71 | |
| 72 | + <%= template_options(Person, 'profile_data') %> | |
| 73 | + | |
| 72 | 74 | <% labelled_fields_for :profile_data, @person do |f| %> | 
| 73 | 75 | <%= render :partial => 'profile_editor/person_form', :locals => {:f => f} %> | 
| 74 | 76 | <% end %> | ... | ... | 
app/views/admin_panel/edit_templates.rhtml
| ... | ... | @@ -1,10 +0,0 @@ | 
| 1 | -<h1><%= _('Edit Templates') %></h1> | |
| 2 | - | |
| 3 | -<ul> | |
| 4 | -<% [[_('Edit Person Template'), environment.person_template], | |
| 5 | - [_('Edit Community Template'), environment.community_template], | |
| 6 | - [__('Edit Enterprise Template'), environment.enterprise_template], | |
| 7 | - [__('Edit Inactive Enterprise Template'), environment.inactive_enterprise_template]].select{|i| i[1]}.each do |row| %> | |
| 8 | -<li><%= link_to row[0], :controller => 'profile_editor', :profile => row[1].identifier %></li> | |
| 9 | -<% end %> | |
| 10 | -</ul> | 
app/views/admin_panel/index.rhtml
| ... | ... | @@ -12,9 +12,10 @@ | 
| 12 | 12 | <tr><td><%= link_to _('Manage User roles'), :controller => 'role' %></td></tr> | 
| 13 | 13 | <tr><td><%= link_to _('Manage users'), :controller => 'users' %></td></tr> | 
| 14 | 14 | <tr><td><%= link_to _('Manage Validators by region'), :controller => 'region_validators' %></td></tr> | 
| 15 | - <tr><td><%= link_to _('Edit Templates'), :action => 'edit_templates' %></td></tr> | |
| 15 | + <tr><td><%= link_to _('Edit Templates'), :controller => 'templates' %></td></tr> | |
| 16 | 16 | <tr><td><%= link_to _('Manage Fields'), :controller => 'features', :action => 'manage_fields' %></td></tr> | 
| 17 | 17 | <tr><td><%= link_to _('Set Portal'), :action => 'set_portal_community' %></td></tr> | 
| 18 | + <tr><td><%= link_to _('Manage Licenses'), :controller =>'licenses' %></td></tr> | |
| 18 | 19 | <% @plugins.dispatch(:admin_panel_links).each do |link| %> | 
| 19 | 20 | <tr><td><%= link_to link[:title], link[:url] %></td></tr> | 
| 20 | 21 | <% end %> | ... | ... | 
| ... | ... | @@ -0,0 +1,9 @@ | 
| 1 | +<% if profile.lat %> | |
| 2 | + <%= block_title block.title %> | |
| 3 | + <div class='the-localization-map'> | |
| 4 | + <img src="http://maps.google.com/maps/api/staticmap?center=<%=profile.lat%>,<%=profile.lng%>&zoom=<%=block.zoom%>&size=190x250&maptype=<%=block.map_type%>&markers=<%=profile.lat%>,<%=profile.lng%>&sensor=false"/> | |
| 5 | + </div> | |
| 6 | +</div> | |
| 7 | +<% else %> | |
| 8 | + <i><%= _('This profile has no geographical position registered.') %></i> | |
| 9 | +<% end %> | ... | ... | 
app/views/catalog/index.rhtml
| ... | ... | @@ -56,7 +56,7 @@ | 
| 56 | 56 | <div> | 
| 57 | 57 | <div class="arrow"></div> | 
| 58 | 58 | <div class="content" id="product-price-composition"> | 
| 59 | - <% product.inputs.each do |i| %> | |
| 59 | + <% product.inputs.relevant_to_price.each do |i| %> | |
| 60 | 60 | <div class="search-product-input-dots-to-price"> | 
| 61 | 61 | <div class="search-product-input-name"><%= i.product_category.name %></div> | 
| 62 | 62 | <%= price_span i.cost, :class => 'search-product-input-price' %> | ... | ... | 
app/views/cms/_blog.rhtml
| ... | ... | @@ -6,6 +6,8 @@ | 
| 6 | 6 | |
| 7 | 7 | <%= required f.text_field(:name, :size => '64', :onchange => "updateUrlField(this, 'article_slug')") %> | 
| 8 | 8 | |
| 9 | +<%= render :partial => 'general_fields' %> | |
| 10 | + | |
| 9 | 11 | <script type="text/javascript"> | 
| 10 | 12 | function submit_button(index) { | 
| 11 | 13 | return $("article_slug").form.select("input.submit")[index]; | ... | ... | 
app/views/cms/_event.rhtml
app/views/cms/_folder.rhtml
app/views/cms/_forum.rhtml
| ... | ... | @@ -6,6 +6,8 @@ | 
| 6 | 6 | |
| 7 | 7 | <%= required f.text_field(:name, :size => '64', :onchange => "updateUrlField(this, 'article_slug')") %> | 
| 8 | 8 | |
| 9 | +<%= render :partial => 'general_fields' %> | |
| 10 | + | |
| 9 | 11 | <%= labelled_form_field(_('Description:'), text_area(:article, :body, :cols => 64, :rows => 10)) %> | 
| 10 | 12 | |
| 11 | 13 | <%= labelled_form_field(_('Posts per page:'), f.select(:posts_per_page, [5, 10, 20, 50, 100])) %> | ... | ... | 
app/views/cms/_gallery.rhtml
| ... | ... | @@ -0,0 +1 @@ | 
| 1 | +<%= labelled_form_field(_('License'), select(:article, :license_id, options_for_select_with_title([[_('None'), nil]] + profile.environment.licenses.map {|license| [license.name, license.id]}, @article.license ? @article.license.id : nil))) %> | ... | ... | 
app/views/cms/_published_article.rhtml
app/views/cms/_raw_html_article.rhtml
app/views/cms/_rss_feed.rhtml
| ... | ... | @@ -2,6 +2,8 @@ | 
| 2 | 2 | |
| 3 | 3 | <%= required f.text_field(:name) %> | 
| 4 | 4 | |
| 5 | +<%= render :partial => 'general_fields' %> | |
| 6 | + | |
| 5 | 7 | <%= required labelled_form_field(_('Limit of articles'), text_field(:article, :limit)) %> | 
| 6 | 8 | |
| 7 | 9 | <%= labelled_form_field(_('Include in RSS Feed only posts from language:'), f.select(:language, [[_('All'), nil ]] + Noosfero.locales.map { |k,v| [v, k]})) %> | ... | ... | 
app/views/cms/_textile_article.rhtml
app/views/cms/_tiny_mce_article.rhtml
| ... | ... | @@ -5,6 +5,7 @@ | 
| 5 | 5 | <div> | 
| 6 | 6 | <%= required labelled_form_field(_('Title'), text_field(:article, 'name', :size => '64')) %> | 
| 7 | 7 | |
| 8 | + <%= render :partial => 'general_fields' %> | |
| 8 | 9 | <%= render :partial => 'translatable' %> | 
| 9 | 10 | <%= render :partial => 'shared/lead_and_body', :locals => {:tiny_mce => true} %> | 
| 10 | 11 | </div> | ... | ... | 
app/views/cms/select_article_type.rhtml
app/views/cms/view.rhtml
| ... | ... | @@ -5,7 +5,7 @@ | 
| 5 | 5 | <% button_bar(:style => 'margin-bottom: 1em;') do %> | 
| 6 | 6 | <% parent_id = ((@article && @article.allow_children?) ? @article : nil) %> | 
| 7 | 7 | |
| 8 | - <%= lightbox_button('new', _('New content'), :action => 'new', :parent_id => parent_id, :cms => true) %> | |
| 8 | + <%= colorbox_button('new', _('New content'), :action => 'new', :parent_id => parent_id, :cms => true) %> | |
| 9 | 9 | <%= button(:back, _('Back to control panel'), :controller => 'profile_editor', :action => "index") %> | 
| 10 | 10 | <% end %> | 
| 11 | 11 | ... | ... | 
| ... | ... | @@ -0,0 +1,22 @@ | 
| 1 | +<%= _('Hi!') %> | |
| 2 | + | |
| 3 | +<%= word_wrap(_('%{sender} (%{sender_link}) commented on the content "%{article_title}".') % { :sender => @sender, :sender_link => url_for(@sender_link), :article_title => @article_title }) %> | |
| 4 | + | |
| 5 | +<%= word_wrap(_('Title: %s') % @comment_title) if @comment_title %> | |
| 6 | + | |
| 7 | +<%= _("Comment:") %> | |
| 8 | +------------------------------------------------------------------------------- | |
| 9 | +<%= word_wrap(@comment_body) %> | |
| 10 | +------------------------------------------------------------------------------- | |
| 11 | + | |
| 12 | +<%= _('Click on the address below to view this comment:') %> | |
| 13 | +<%= url_for @comment_url %> | |
| 14 | + | |
| 15 | +<%= _('Click on the address below to cancel the notification of new comments:') %> | |
| 16 | +<%= url_for @unsubscribe_url %> | |
| 17 | + | |
| 18 | +<%= _("Greetings,") %> | |
| 19 | + | |
| 20 | +-- | |
| 21 | +<%= _('%s team.') % @environment %> | |
| 22 | +<%= url_for @url %> | ... | ... | 
app/views/content_viewer/_article_toolbar.rhtml
| ... | ... | @@ -34,7 +34,7 @@ | 
| 34 | 34 | <%= expirable_button @page, :locale, content, url %> | 
| 35 | 35 | <% end %> | 
| 36 | 36 | |
| 37 | - <%= lightbox_remote_button(:new, label_for_new_article(@page), profile.admin_url.merge(:controller => 'cms', :action => 'new', :parent_id => (@page.folder? ? @page : (@page.parent.nil? ? nil : @page.parent)))) %> | |
| 37 | + <%= colorbox_button(:new, label_for_new_article(@page), profile.admin_url.merge(:controller => 'cms', :action => 'new', :parent_id => (@page.folder? ? @page : (@page.parent.nil? ? nil : @page.parent)))) %> | |
| 38 | 38 | <% end %> | 
| 39 | 39 | |
| 40 | 40 | <% if @page.accept_uploads? && @page.allow_create?(user) %> | ... | ... | 
| ... | ... | @@ -0,0 +1,13 @@ | 
| 1 | +<% if @unfollow_form %> | |
| 2 | +<div class='unfollow-article'> | |
| 3 | + <h1><%= _('Cancel notification of new comments') %></h1> | |
| 4 | + <p><%= _("Fill in the following field with your e-mail if you don't want to be notified when this content receives new comments anymore.") %></p> | |
| 5 | + <% form_tag(@page.view_url.merge({:only_path => true}), {:method => 'post', :class => 'comment_form'}) do %> | |
| 6 | + <%= hidden_field_tag(:unfollow, 'commit') %> | |
| 7 | + <%= labelled_form_field(_('Enter your e-Mail'), text_field_tag(:email, nil, {:size => 40})) %> | |
| 8 | + <% button_bar do %> | |
| 9 | + <%= submit_button(:ok, _('Cancel notifications for e-mail above') ) %> | |
| 10 | + <% end %> | |
| 11 | + <% end %> | |
| 12 | +</div> | |
| 13 | +<% end %> | ... | ... | 
app/views/content_viewer/view_page.rhtml
| ... | ... | @@ -6,12 +6,30 @@ | 
| 6 | 6 | |
| 7 | 7 | <div id="article" class="<%= @page.css_class_name %>"> | 
| 8 | 8 | |
| 9 | +<%= render :partial => 'confirm_unfollow' %> | |
| 10 | + | |
| 9 | 11 | <div id="article-toolbar"></div> | 
| 10 | 12 | |
| 11 | 13 | <script type="text/javascript"> | 
| 12 | - <%= remote_function :update => "article-toolbar", :url => @page.url.merge({ :toolbar => true, :only_path => true }), :complete => "$$('#article-toolbar .remote-lbOn').each(function(link) { new lightbox(link); }); jQuery('#article-toolbar .simplemenu-trigger').click(function(e) { e.stopPropagation(); })" %> | |
| 14 | + <%= remote_function :update => "article-toolbar", :url => @page.url.merge({ :toolbar => true, :only_path => true }) %> | |
| 13 | 15 | </script> | 
| 14 | 16 | |
| 17 | +<% if @page.display_hits? || @page.license.present? %> | |
| 18 | + <div id='article-sub-header'> | |
| 19 | + <% if @page.display_hits? %> | |
| 20 | + <div id="article-hits"> | |
| 21 | + <%= n_('Viewed one time', 'Viewed %{num} times', @page.hits) % { :num => @page.hits } %> | |
| 22 | + </div> | |
| 23 | + <% end %> | |
| 24 | + | |
| 25 | + <% if @page.license.present? %> | |
| 26 | + <div id="article-license"> | |
| 27 | + <%= _('Licensed under %s') % (@page.license.url.present? ? link_to(@page.license.name, @page.license.url, :target => '_blank') : @page.license.name) %> | |
| 28 | + </div> | |
| 29 | + <% end %> | |
| 30 | + </div> | |
| 31 | +<% end %> | |
| 32 | + | |
| 15 | 33 | <% if !@page.tags.empty? %> | 
| 16 | 34 | <div id="article-tags"> | 
| 17 | 35 | <%= _("This article's tags:") %> | 
| ... | ... | @@ -19,12 +37,6 @@ | 
| 19 | 37 | </div> | 
| 20 | 38 | <% end %> | 
| 21 | 39 | |
| 22 | -<% if @page.display_hits? %> | |
| 23 | - <div id="article-hits"> | |
| 24 | - <%= n_('Viewed one time', 'Viewed %{num} times', @page.hits) % { :num => @page.hits } %> | |
| 25 | - </div> | |
| 26 | -<% end %> | |
| 27 | - | |
| 28 | 40 | <% if @page.parent && !@page.parent.path.blank? %> | 
| 29 | 41 | <div id="article-parent"> | 
| 30 | 42 | <%= button(:back, _('Go back to %s') % @page.parent.short_title, @page.parent.url) %> | ... | ... | 
app/views/edit_template/index.rhtml
| ... | ... | @@ -1,14 +0,0 @@ | 
| 1 | -mexendo o contedudo para ver como fica a parada toda | |
| 2 | -mexendo o contedudo para ver como fica a parada toda | |
| 3 | -mexendo o contedudo para ver como fica a parada toda | |
| 4 | -mexendo o contedudo para ver como fica a parada toda | |
| 5 | -mexendo o contedudo para ver como fica a parada toda | |
| 6 | -mexendo o contedudo para ver como fica a parada toda | |
| 7 | -mexendo o contedudo para ver como fica a parada toda | |
| 8 | -mexendo o contedudo para ver como fica a parada toda | |
| 9 | -mexendo o contedudo para ver como fica a parada toda | |
| 10 | -mexendo o contedudo para ver como fica a parada toda | |
| 11 | -mexendo o contedudo para ver como fica a parada toda | |
| 12 | -mexendo o contedudo para ver como fica a parada toda | |
| 13 | -mexendo o contedudo para ver como fica a parada toda | |
| 14 | - | 
app/views/enterprise_registration/basic_information.rhtml
| ... | ... | @@ -28,6 +28,8 @@ | 
| 28 | 28 | <%= hidden_field_tag 'create_enterprise[target_id]', environment.id %> | 
| 29 | 29 | <% end %> | 
| 30 | 30 | |
| 31 | + <%= template_options(Enterprise, 'create_enterprise')%> | |
| 32 | + | |
| 31 | 33 | <% button_bar do %> | 
| 32 | 34 | <%= submit_button('next', _('Next'), :cancel => {:profile => current_user.person.identifier, :action=>"enterprises", :controller=>"profile"}) %> | 
| 33 | 35 | <% end %> | ... | ... | 
| ... | ... | @@ -0,0 +1,12 @@ | 
| 1 | +<%= error_messages_for :license %> | |
| 2 | + | |
| 3 | +<% form_for :license, @license do |f| %> | |
| 4 | + <%= hidden_field_tag(:license_id, params[:license_id]) %> | |
| 5 | + <%= required labelled_form_field(_('Name'), f.text_field(:name)) %> | |
| 6 | + <%= labelled_form_field(_('License url'), f.text_field(:url)) %> | |
| 7 | + | |
| 8 | + <% button_bar do %> | |
| 9 | + <%= submit_button('save', _('Save'))%> | |
| 10 | + <%= button('cancel', _('Cancel'), {:action => 'index'})%> | |
| 11 | + <% end %> | |
| 12 | +<% end %> | ... | ... | 
| ... | ... | @@ -0,0 +1,22 @@ | 
| 1 | +<h1><%= _('Manage licenses') %></h1> | |
| 2 | +<table style='overflow: hidden;'> | |
| 3 | + <tr> | |
| 4 | + <th style='width: 25%'><%= _('Name') %></th> | |
| 5 | + <th style='width: 60%'><%= _('Url reference') %></th> | |
| 6 | + <th style='width: 15%'><%= _('Actions') %></th> | |
| 7 | + </tr> | |
| 8 | + <% @licenses.each do |license| %> | |
| 9 | + <tr> | |
| 10 | + <td title="<%= license.name%>"><%= truncate(license.name, :length => 19) %></td> | |
| 11 | + <td title="<%= license.url %>"><%= license.url.present? ? link_to(truncate(license.url, :length => 60), license.url, :target => '_blank') : '' %></td> | |
| 12 | + <td style='white-space: nowrap;'> | |
| 13 | + <%= button_without_text :edit, _('Edit'), :action => 'edit', :license_id => license.id %> | |
| 14 | + <%= button_without_text :remove, _('Remove'), {:action => 'remove', :license_id => license.id}, :method => 'post', :confirm => _('Are you sure you want to remove this license?') %></td> | |
| 15 | + </tr> | |
| 16 | + <% end %> | |
| 17 | +</table> | |
| 18 | + | |
| 19 | +<% button_bar do %> | |
| 20 | + <%= button(:add, _('Add a new license'), :action => 'create')%> | |
| 21 | + <%= button :back, _('Back to admin panel'), :controller => 'admin_panel' %> | |
| 22 | +<% end %> | ... | ... | 
app/views/maps/_google_map.js.erb
| ... | ... | @@ -5,27 +5,113 @@ var marker; | 
| 5 | 5 | var center; | 
| 6 | 6 | var move = true; | 
| 7 | 7 | var previousCenter; | 
| 8 | +var mapZoom = 15; | |
| 9 | +var delay_autocomplete = 500; | |
| 8 | 10 | |
| 9 | -function getAddress(latlng) { | |
| 11 | +function pointToAddress(latlng) { | |
| 10 | 12 | $('location-fields').addClassName("loading"); | 
| 11 | 13 | |
| 12 | - if (latlng != null) { | |
| 13 | - geocoder.geocode( {'latLng': latlng}, showAddress); | |
| 14 | - } | |
| 14 | + if (latlng == null) | |
| 15 | + return; | |
| 16 | + | |
| 17 | + geocoder.geocode( {'latLng': latlng}, function(results, status) { | |
| 18 | + if (status != google.maps.GeocoderStatus.OK) { | |
| 19 | + alert("<%=_("Address not found, reason:")%>" + statusErrorMessage(status)); | |
| 20 | + return; | |
| 21 | + } | |
| 22 | + | |
| 23 | + var place = results[0]; | |
| 24 | + | |
| 25 | + $('location-fields').removeClassName("loading"); | |
| 26 | + | |
| 27 | + var position = marker.getPosition(); | |
| 28 | + $('profile_data_lat').value = position.lat(); | |
| 29 | + $('profile_data_lng').value = position.lng(); | |
| 30 | + | |
| 31 | + form = jQuery('#location-form')[0]; | |
| 32 | + form.lat = marker.getPosition().lat(); | |
| 33 | + form.lng = marker.getPosition().lng(); | |
| 34 | + | |
| 35 | + var components_len = place.address_components.size(); | |
| 36 | + if (components_len < 2) | |
| 37 | + return; | |
| 38 | + | |
| 39 | + var country_code = ""; | |
| 40 | + var state = ""; | |
| 41 | + var city = ""; | |
| 42 | + var zip_code = ""; | |
| 43 | + var route = ""; | |
| 44 | + var number = ""; | |
| 45 | + var sublocality = ""; | |
| 46 | + var address = ""; | |
| 47 | + | |
| 48 | + var i; | |
| 49 | + var has_postal_code = false; | |
| 50 | + for (i=0; i < components_len; i++) { | |
| 51 | + type = place.address_components[i].types[0]; | |
| 52 | + if (type == 'postal_code') | |
| 53 | + has_postal_code = true; | |
| 54 | + } | |
| 55 | + | |
| 56 | + for (i=0; i < components_len; i++) { | |
| 57 | + type = place.address_components[i].types[0]; | |
| 58 | + value = place.address_components[i]; | |
| 59 | + | |
| 60 | + if (type == 'country') | |
| 61 | + country_code = value.short_name; | |
| 62 | + else if (type == 'administrative_area_level_1') | |
| 63 | + state = value.long_name; | |
| 64 | + else if (type == 'locality') | |
| 65 | + city = value.long_name; | |
| 66 | + else if (type == 'postal_code') | |
| 67 | + zip_code = value.short_name; | |
| 68 | + if (has_postal_code) { | |
| 69 | + if (type == "route") | |
| 70 | + route = value.long_name; | |
| 71 | + else if (type == "street_number") | |
| 72 | + number = value.short_name; | |
| 73 | + else if (type == 'sublocality') | |
| 74 | + sublocality = value.long_name; | |
| 75 | + } | |
| 76 | + } | |
| 77 | + | |
| 78 | + // build address | |
| 79 | + if (route) { | |
| 80 | + address = route; | |
| 81 | + if (number) | |
| 82 | + address = address + ', ' + number; | |
| 83 | + if (sublocality && sublocality != city) | |
| 84 | + address = address + ', ' + sublocality; | |
| 85 | + } | |
| 86 | + | |
| 87 | + if (country_code) | |
| 88 | + $('profile_data_country').value = country_code; | |
| 89 | + if (state) | |
| 90 | + $('profile_data_state').value = state; | |
| 91 | + if (city) | |
| 92 | + $('profile_data_city').value = city; | |
| 93 | + if (zip_code) | |
| 94 | + $('profile_data_zip_code').value = zip_code; | |
| 95 | + if (address) | |
| 96 | + $('profile_data_address').value = address; | |
| 97 | + | |
| 98 | + map.setCenter(marker.getPosition()); | |
| 99 | + }); | |
| 15 | 100 | } | 
| 16 | 101 | |
| 17 | -function codeAddress() { | |
| 102 | +function addressToPoint() { | |
| 18 | 103 | $('location-fields').addClassName("loading"); | 
| 19 | 104 | |
| 20 | 105 | var country_option = $('profile_data_country').value; | 
| 21 | - var address = $('profile_data_address').value + "-" + $('profile_data_zip_code').value + "," + $('profile_data_city').value+ "-" + $('profile_data_state').value + "," + country_option; | |
| 106 | + var address = $('profile_data_address').value + ", " + $('profile_data_zip_code').value + ", " | |
| 107 | + + $('profile_data_city').value+ ", " + $('profile_data_state').value + ", " + country_option; | |
| 22 | 108 | |
| 23 | 109 | if (geocoder) { | 
| 24 | - geocoder.geocode( { 'address': address}, function(results, status) { | |
| 110 | + geocoder.geocode({ 'address': address}, function(results, status) { | |
| 25 | 111 | if (status == google.maps.GeocoderStatus.OK) { | 
| 26 | 112 | map.setCenter(results[0].geometry.location); | 
| 27 | 113 | marker.setPosition(results[0].geometry.location); | 
| 28 | - getAddress(marker.getPosition()); | |
| 114 | + pointToAddress(marker.getPosition()); | |
| 29 | 115 | |
| 30 | 116 | $('profile_data_lat').value = results[0].geometry.location.lat(); | 
| 31 | 117 | $('profile_data_lng').value = results[0].geometry.location.lng(); | 
| ... | ... | @@ -33,17 +119,15 @@ function codeAddress() { | 
| 33 | 119 | enable_save(); | 
| 34 | 120 | } else { | 
| 35 | 121 | $('location-fields').removeClassName("loading"); | 
| 36 | - alert('<%=_("Address not found, reason:")%>' + translate_status(status)); | |
| 122 | + alert('<%=_("Address not found, reason:")%>' + statusErrorMessage(status)); | |
| 37 | 123 | } | 
| 38 | 124 | }); | 
| 39 | 125 | } | 
| 40 | 126 | |
| 41 | - map.setZoom(11); | |
| 42 | - | |
| 43 | 127 | return false; | 
| 44 | 128 | } | 
| 45 | 129 | |
| 46 | -function translate_status(status) | |
| 130 | +function statusErrorMessage(status) | |
| 47 | 131 | { | 
| 48 | 132 | var translated_status = ''; | 
| 49 | 133 | |
| ... | ... | @@ -59,136 +143,42 @@ function translate_status(status) | 
| 59 | 143 | return translated_status; | 
| 60 | 144 | } | 
| 61 | 145 | |
| 62 | -function getAddressData() { | |
| 63 | - var text = ''; | |
| 64 | - var fields = [ | |
| 65 | - 'profile_data_country', | |
| 66 | - 'profile_data_state', | |
| 67 | - 'profile_data_city', | |
| 68 | - 'profile_data_address', | |
| 69 | - 'profile_data_zip_code' | |
| 70 | - ]; | |
| 71 | - for (var i = 0; i < fields.length; i++) { | |
| 72 | - var field = fields[i]; | |
| 73 | - if ($(field)) { | |
| 74 | - text += $(field).value + " "; | |
| 75 | - } | |
| 76 | - } | |
| 77 | - return text; | |
| 78 | -} | |
| 79 | - | |
| 80 | -function showAddress(results, status) { | |
| 81 | - | |
| 82 | - if (status == google.maps.GeocoderStatus.OK) { | |
| 83 | - map.setCenter(results[0].geometry.location); | |
| 84 | - updateFields(results[0]); | |
| 85 | - | |
| 86 | - } else { | |
| 87 | - alert("<%=_("Address not found, reason:")%>" + translate_status(status)); | |
| 88 | - } | |
| 89 | - | |
| 90 | -} | |
| 91 | - | |
| 92 | -function updateFields(place) { | |
| 93 | - var position = marker.getPosition(); | |
| 94 | - $('profile_data_lat').value = position.lat(); | |
| 95 | - $('profile_data_lng').value = position.lng(); | |
| 96 | - $('location-fields').removeClassName("loading"); | |
| 97 | - | |
| 98 | - form = jQuery('#location-form')[0]; | |
| 99 | - form.lat = marker.getPosition().lat(); | |
| 100 | - form.lng = marker.getPosition().lng(); | |
| 101 | - | |
| 102 | - var components_len = place.address_components.size(); | |
| 103 | - | |
| 104 | - if(components_len < 2) | |
| 105 | - { | |
| 106 | - return false; | |
| 107 | - } | |
| 108 | - | |
| 109 | - var components = place.address_components; | |
| 110 | - var address = ""; | |
| 111 | - var zip_code = ""; | |
| 112 | - var city = ""; | |
| 113 | - var state = ""; | |
| 114 | - var country_code = ""; | |
| 115 | - var i = 0; | |
| 116 | - | |
| 117 | - for( i =0 ; i < components_len; i ++) | |
| 118 | - { | |
| 119 | - | |
| 120 | - if (components[i].types[0] == 'country') | |
| 121 | - country_code = components[i].short_name; | |
| 122 | - else if (components[i].types[0] == 'administrative_area_level_1') | |
| 123 | - state = components[i].long_name; | |
| 124 | - else if (components[i].types[0] == 'locality') | |
| 125 | - city = components[i].long_name; | |
| 126 | - else if (components[i].types[0] == 'sublocality') | |
| 127 | - address = components[i].long_name + "-" + address; | |
| 128 | - else if (components[i].types[0] == "route") | |
| 129 | - address = components[i].long_name + address; | |
| 130 | - else if (components[i].types[0] == "street_number") | |
| 131 | - address = address + "," + components[i].short_name ; | |
| 132 | - else if (components[i].types[0] == 'postal_code') | |
| 133 | - zip_code = components[i].short_name; | |
| 134 | - } | |
| 135 | - | |
| 136 | - $('profile_data_country').value = country_code; | |
| 137 | - $('profile_data_state').value = state; | |
| 138 | - $('profile_data_address').value = address; | |
| 139 | - $('profile_data_city').value = city; | |
| 140 | - $('profile_data_zip_code').value = zip_code; | |
| 141 | -} | |
| 142 | - | |
| 143 | - | |
| 144 | -function initialize_map() { | |
| 146 | +function initializeMap() { | |
| 145 | 147 | geocoder = new google.maps.Geocoder(); | 
| 146 | 148 | |
| 147 | 149 | var lat = <%= profile.lat || 'false' %>; | 
| 148 | 150 | var lng = <%= profile.lng || 'false' %>; | 
| 149 | - | |
| 150 | 151 | if ( !(lat && lng) ) { | 
| 151 | 152 | lat = -15.7605361485013; | 
| 152 | 153 | lng = -47.933349609375; | 
| 153 | 154 | } | 
| 154 | 155 | |
| 155 | - var latlng = new google.maps.LatLng(lat,lng); | |
| 156 | - | |
| 157 | - var myOptions = { | |
| 158 | - zoom: 8, | |
| 159 | - center: latlng, | |
| 160 | - mapTypeId: google.maps.MapTypeId.ROADMAP | |
| 161 | - } | |
| 162 | - | |
| 163 | - center = latlng; | |
| 164 | - | |
| 165 | - map = new google.maps.Map(document.getElementById("location-map"), myOptions); | |
| 166 | - | |
| 167 | - continueLoadMapV3() | |
| 168 | -} | |
| 169 | - | |
| 170 | -function continueLoadMapV3() { | |
| 156 | + var center = new google.maps.LatLng(lat,lng);; | |
| 157 | + map = new google.maps.Map(document.getElementById("location-map"), { | |
| 158 | + zoom: mapZoom, | |
| 159 | + center: center, | |
| 160 | + mapTypeId: google.maps.MapTypeId.HYBRID | |
| 161 | + }); | |
| 171 | 162 | |
| 172 | - marker = new google.maps.Marker({ | |
| 173 | - position: center, | |
| 174 | - map: map, | |
| 175 | - draggable: true | |
| 176 | - }); | |
| 163 | + marker = new google.maps.Marker({ | |
| 164 | + position: center, | |
| 165 | + map: map, | |
| 166 | + draggable: true | |
| 167 | + }); | |
| 177 | 168 | |
| 178 | - google.maps.event.addListener(marker, "dragend", function() { | |
| 179 | - move = false; | |
| 180 | - getAddress(marker.getPosition()); | |
| 181 | - enable_save(); | |
| 182 | - }); | |
| 169 | + google.maps.event.addListener(marker, "dragend", function() { | |
| 170 | + move = false; | |
| 171 | + pointToAddress(marker.getPosition()); | |
| 172 | + map.setCenter(marker.getPosition()); | |
| 173 | + enable_save(); | |
| 174 | + }); | |
| 183 | 175 | |
| 184 | 176 | } | 
| 185 | 177 | |
| 186 | -window.onload = initialize_map; | |
| 187 | - | |
| 188 | -var delay_autocomplete = 500; | |
| 189 | - | |
| 190 | 178 | jQuery.noConflict(); | 
| 191 | -jQuery(document).ready(function (){ | |
| 179 | +jQuery(document).ready(function () { | |
| 180 | + | |
| 181 | + initializeMap(); | |
| 192 | 182 | |
| 193 | 183 | jQuery.widget( "custom.catcomplete",jQuery.ui.autocomplete, { | 
| 194 | 184 | _renderMenu: function( ul, items ) { | 
| ... | ... | @@ -204,7 +194,6 @@ jQuery(document).ready(function (){ | 
| 204 | 194 | } | 
| 205 | 195 | }); | 
| 206 | 196 | |
| 207 | - | |
| 208 | 197 | jQuery("#profile_data_city").catcomplete({ | 
| 209 | 198 | source: "../maps/search_city", | 
| 210 | 199 | minLength: 3, | 
| ... | ... | @@ -219,21 +208,13 @@ jQuery(document).ready(function (){ | 
| 219 | 208 | }); | 
| 220 | 209 | |
| 221 | 210 | jQuery("#profile_data_city").keyup(function(){ | 
| 222 | - | |
| 223 | 211 | disable_save(); | 
| 224 | - | |
| 225 | 212 | }); | 
| 226 | - | |
| 227 | 213 | jQuery("#profile_data_state").keyup(function(){ | 
| 228 | - | |
| 229 | 214 | disable_save(); | 
| 230 | - | |
| 231 | 215 | }); | 
| 232 | - | |
| 233 | 216 | jQuery("#profile_data_country").change(function(){ | 
| 234 | - | |
| 235 | 217 | disable_save(); | 
| 236 | - | |
| 237 | 218 | }); | 
| 238 | 219 | |
| 239 | 220 | }); | ... | ... | 
app/views/maps/edit_location.rhtml
| ... | ... | @@ -12,7 +12,7 @@ | 
| 12 | 12 | <%= labelled_form_field _('ZIP code'), text_field(:profile_data, :zip_code) %> | 
| 13 | 13 | <%= labelled_form_field _('Address (street and number)'), text_field(:profile_data, :address) %> | 
| 14 | 14 | <% button_bar do %> | 
| 15 | - <%= button_to_function :search, _('Locate in the map'), "codeAddress()", :title => _("Locate the address informed above in the map below (note that you'll probably need to adjust the marker to get a precise position)") %> | |
| 15 | + <%= button_to_function :search, _('Locate in the map'), "addressToPoint()", :title => _("Locate the address informed above in the map below (note that you'll probably need to adjust the marker to get a precise position)") %> | |
| 16 | 16 | <%= submit_button 'save', _('Save') %> | 
| 17 | 17 | <%= button(:back, _('Back to control panel'), :controller => 'profile_editor') %> | 
| 18 | 18 | <% end %> | ... | ... | 
app/views/memberships/new_community.rhtml
app/views/profile/_profile_comment_form.rhtml
| ... | ... | @@ -3,15 +3,17 @@ | 
| 3 | 3 | <p class='profile-wall-reply'> | 
| 4 | 4 | <% update_area = tab_action == 'wall' ? 'profile_activities' : 'network-activities' %> | 
| 5 | 5 | <% form_remote_tag :url => {:controller => 'profile', :action => 'leave_comment_on_activity', :tab_action => tab_action}, :html => { :class => 'profile-wall-reply-form', 'data-update' => update_area } do %> | 
| 6 | - <%= text_area :comment, :body, {:id => "reply_content_#{activity.id}", | |
| 7 | - :cols => 50, | |
| 8 | - :rows => 1, | |
| 9 | - :class => 'submit-with-keypress', | |
| 10 | - :title => _('Leave your comment'), | |
| 11 | - :onfocus => ('if(this.value==this.title){this.value="";this.style.color="#000"};this.style.backgroundImage="url(' + profile_icon(current_person, :icon, false) + ')"' if logged_in?), | |
| 12 | - :onblur => ('if(this.value==""){this.value=this.title;this.style.color="#ccc"};this.style.backgroundImage="none"' if logged_in?), | |
| 13 | - :value => _('Leave your comment'), | |
| 14 | - :style => 'color: #ccc' } %> | |
| 6 | + <%= expandable_text_area :comment, | |
| 7 | + :body, | |
| 8 | + "reply_content_#{activity.id}", | |
| 9 | + :cols => 50, | |
| 10 | + :rows => 1, | |
| 11 | + :class => 'submit-with-keypress', | |
| 12 | + :title => _('Leave your comment'), | |
| 13 | + :onfocus => ('if(this.value==this.title){this.value="";this.style.color="#000"};this.style.backgroundImage="url(' + profile_icon(current_person, :icon, false) + ')"' if logged_in?), | |
| 14 | + :onblur => ('if(this.value==""){this.value=this.title;this.style.color="#ccc"};this.style.backgroundImage="none"' if logged_in?), | |
| 15 | + :value => _('Leave your comment'), | |
| 16 | + :style => 'color: #ccc' %> | |
| 15 | 17 | <%= hidden_field_tag :source_id, activity.id, :id => "activity_id_#{activity.id}" %> | 
| 16 | 18 | <% end %> | 
| 17 | 19 | </p> | ... | ... | 
app/views/profile/_profile_scrap_reply_form.rhtml
| ... | ... | @@ -2,15 +2,16 @@ | 
| 2 | 2 | <div id='profile-wall-reply-form-<%= scrap.id%>' style='display:none'> | 
| 3 | 3 | <p class='profile-wall-reply'> | 
| 4 | 4 | <% form_remote_tag :url => {:controller => 'profile', :action => 'leave_scrap'}, :update => "profile_activities", :html => { :class => 'profile-wall-reply-form'} do %> | 
| 5 | - <%= text_area :scrap, :content, { :id => "reply_content_#{scrap.id}", | |
| 6 | - :cols => 50, | |
| 7 | - :rows => 1, | |
| 8 | - :class => 'submit-with-keypress', | |
| 9 | - :title => _('Leave your comment'), | |
| 10 | - :onfocus => ('if(this.value==this.title){this.value="";this.style.color="#000"};this.style.backgroundImage="url(' + profile_icon(current_person, :icon, false) + ')"' if logged_in?), | |
| 11 | - :onblur => ('if(this.value==""){this.value=this.title;this.style.color="#ccc"};this.style.backgroundImage="none"' if logged_in?), | |
| 12 | - :value => _('Leave your comment') | |
| 13 | - } %> | |
| 5 | + <%= expandable_text_area :scrap, | |
| 6 | + :content, | |
| 7 | + "reply_content_#{scrap.id}", | |
| 8 | + :cols => 50, | |
| 9 | + :rows => 1, | |
| 10 | + :class => 'submit-with-keypress', | |
| 11 | + :title => _('Leave your comment'), | |
| 12 | + :onfocus => ('if(this.value==this.title){this.value="";this.style.color="#000"};this.style.backgroundImage="url(' + profile_icon(current_person, :icon, false) + ')"' if logged_in?), | |
| 13 | + :onblur => ('if(this.value==""){this.value=this.title;this.style.color="#ccc"};this.style.backgroundImage="none"' if logged_in?), | |
| 14 | + :value => _('Leave your comment') %> | |
| 14 | 15 | <%= hidden_field_tag 'scrap[scrap_id]', scrap.id %> | 
| 15 | 16 | <%= hidden_field_tag 'receiver_id', scrap.sender.id %> | 
| 16 | 17 | <% end %> | ... | ... | 
app/views/profile/_upload_image.rhtml
| ... | ... | @@ -6,15 +6,9 @@ | 
| 6 | 6 | <p class='profile-activity-text'><%= link_to activity.user.name, activity.user.url %> <%= describe activity %></p> | 
| 7 | 7 | <p class='profile-activity-time'><%= time_ago_as_sentence(activity.created_at) %></p> | 
| 8 | 8 | <div class='profile-wall-actions'> | 
| 9 | - <%= link_to(s_('profile|Comment'), '#', { :class => 'focus-on-comment'}) unless activity.get_view_url.size == 1 %> | |
| 10 | 9 | <%= link_to_remote(content_tag(:span, _('Remove')), :confirm => _('Are you sure?'), :url =>{:action => 'remove_activity', :activity_id => activity.id}, :update => "profile-activity-item-#{activity.id}") if logged_in? && current_person == @profile %> | 
| 11 | 10 | </div> | 
| 12 | 11 | </div> | 
| 13 | 12 | </div> | 
| 14 | 13 | <div title='<%= activity.target.class.short_description %>' class='profile-activity-icon icon-new icon-newgallery'></div> | 
| 15 | - | |
| 16 | -<% if activity.get_view_url.size == 1 %> | |
| 17 | - <%= render :partial => 'profile_comments', :locals => { :activity => activity, :tab_action => tab_action } %> | |
| 18 | -<% end %> | |
| 19 | - | |
| 20 | 14 | <br/> | ... | ... | 
app/views/profile/index.rhtml
| ... | ... | @@ -29,7 +29,6 @@ | 
| 29 | 29 | <% if logged_in? && current_person.follows?(@profile) %> | 
| 30 | 30 | <% tabs << {:title => _('Wall'), :id => 'profile-wall', :content => (render :partial => 'profile_wall')} %> | 
| 31 | 31 | <% end %> | 
| 32 | - <% tabs << {:title => _('What\'s new'), :id => 'profile-network', :content => (render :partial => 'profile_network')} %> | |
| 33 | 32 | <% elsif @profile.person? %> | 
| 34 | 33 | <% tabs << {:title => _('Profile'), :id => 'person-profile', :content => (render :partial => 'person_profile')} %> | 
| 35 | 34 | <% if logged_in? && current_person.follows?(@profile) %> | ... | ... | 
app/views/profile_editor/edit.rhtml
| ... | ... | @@ -4,6 +4,12 @@ | 
| 4 | 4 | |
| 5 | 5 | <% labelled_form_for :profile_data, @profile, :html => { :id => 'profile-data', :multipart => true } do |f| %> | 
| 6 | 6 | |
| 7 | + <% if user.has_permission?('manage_environment_templates', profile.environment) %> | |
| 8 | + <div id="profile-is-template"> | |
| 9 | + <%= labelled_check_box(_('This profile is a template'), 'profile_data[is_template]', true, @profile.is_template) %> | |
| 10 | + </div> | |
| 11 | + <% end %> | |
| 12 | + | |
| 7 | 13 | <%= render :partial => partial_for_class(@profile.class), :locals => { :f => f } %> | 
| 8 | 14 | |
| 9 | 15 | <% unless @profile.person? && @environment.active_person_fields.include?('image') %> | ... | ... | 
app/views/search/_product.rhtml
| ... | ... | @@ -27,9 +27,9 @@ | 
| 27 | 27 | <% end %> | 
| 28 | 28 | |
| 29 | 29 | <% if product.price_described? %> | 
| 30 | - <% title = (product.inputs + product.price_details).map{ |i| | |
| 30 | + <% title = (product.inputs.relevant_to_price + product.price_details).map{ |i| | |
| 31 | 31 | '<div class="search-product-input-dots-to-price">' + | 
| 32 | - '<div class="search-product-input-name">' + i.product_category.name + '</div>' + | |
| 32 | + '<div class="search-product-input-name">' + i.name + '</div>' + | |
| 33 | 33 | price_span(i.price, :class => 'search-product-input-price') + | 
| 34 | 34 | '</div>' }.join('') %> | 
| 35 | 35 | <%= link_to_function _("Open Price"), '', :title => title, :class => "search-product-price-details" %> | ... | ... | 
| ... | ... | @@ -0,0 +1,19 @@ | 
| 1 | +<% if @error %> | |
| 2 | + <div class="errorExplanation" id="errorExplanation"> | |
| 3 | + <h2><%= _('The template could not be saved') %></h2> | |
| 4 | + <p><%= _('There were problems with the following fields:') %> </p> | |
| 5 | + <ul> | |
| 6 | + <li><%= @error %></li> | |
| 7 | + </ul> | |
| 8 | + </div> | |
| 9 | +<% end %> | |
| 10 | + | |
| 11 | +<% form_tag do %> | |
| 12 | + <%= labelled_text_field(_('Name')+': ', :name)%> | |
| 13 | + | |
| 14 | + <% button_bar do %> | |
| 15 | + <%= submit_button('save', _('Save'))%> | |
| 16 | + <%= button('cancel', _('Cancel'), {:action => 'index'})%> | |
| 17 | + <% end %> | |
| 18 | +<% end %> | |
| 19 | + | ... | ... | 
| ... | ... | @@ -0,0 +1,12 @@ | 
| 1 | +<% title = case @kind | |
| 2 | + when 'person' | |
| 3 | + | |
| 4 | + when 'community' | |
| 5 | + _('Create community template') | |
| 6 | + when 'enterprise' | |
| 7 | + _('Create enterprise template') | |
| 8 | +end %> | |
| 9 | + | |
| 10 | +<h1><%= _('Create person template') %></h1> | |
| 11 | + | |
| 12 | +<%= render :partial => 'create_template_form' %> | ... | ... | 
| ... | ... | @@ -0,0 +1,26 @@ | 
| 1 | +<h1><%= _('Edit Templates') %></h1> | |
| 2 | + | |
| 3 | +<%= _('Manage the templates used on creation of profiles') %> | |
| 4 | + | |
| 5 | +<% list_of_templates = [[_('Person') , Person.templates , 'person' ], | |
| 6 | + [_('Community') , Community.templates , 'community' ], | |
| 7 | + [_('Enterprise'), Enterprise.templates, 'enterprise']] %> | |
| 8 | + | |
| 9 | +<% list_of_templates.each do |title, templates, kind|%> | |
| 10 | + <div class='template-kind'> | |
| 11 | + <h2><%= title %></h2> | |
| 12 | + <%= button :add, _('New...'), {:action => "create_#{kind}_template"}, :title => _("Create a new template for %s") % title.downcase %> | |
| 13 | + <ul> | |
| 14 | + <% templates.each do |template| %> | |
| 15 | + <li> | |
| 16 | + <%= image_tag "icons-app/#{kind}-icon.png" %> | |
| 17 | + <%= link_to(template.name, {:controller => 'profile_editor', :profile => template.identifier}, :title => _('Edit template "%s"') % template.name ) %> | |
| 18 | + </li> | |
| 19 | + <% end %> | |
| 20 | + </ul> | |
| 21 | + </div> | |
| 22 | +<% end %> | |
| 23 | + | |
| 24 | +<% button_bar do %> | |
| 25 | + <%= button :back, _('Back to admin panel'), :controller => 'admin_panel' %> | |
| 26 | +<% end %> | ... | ... | 
db/migrate/20120710033223_add_template_and_is_template_fields.rb
0 → 100644
| ... | ... | @@ -0,0 +1,11 @@ | 
| 1 | +class AddTemplateAndIsTemplateFields < ActiveRecord::Migration | |
| 2 | + def self.up | |
| 3 | + add_column :profiles, :is_template, :boolean, :default => false | |
| 4 | + add_column :profiles, :template_id, :integer | |
| 5 | + end | |
| 6 | + | |
| 7 | + def self.down | |
| 8 | + remove_column :profiles, :is_template | |
| 9 | + remove_column :profiles, :template_id | |
| 10 | + end | |
| 11 | +end | ... | ... | 
db/migrate/20120710062802_fill_is_template_field_on_basic_templates.rb
0 → 100644
| ... | ... | @@ -0,0 +1,14 @@ | 
| 1 | +class CreateLicenses < ActiveRecord::Migration | |
| 2 | + def self.up | |
| 3 | + create_table :licenses do |t| | |
| 4 | + t.string :name, :null => false | |
| 5 | + t.string :slug, :null => false | |
| 6 | + t.string :url | |
| 7 | + t.references :environment, :null => false | |
| 8 | + end | |
| 9 | + end | |
| 10 | + | |
| 11 | + def self.down | |
| 12 | + drop_table :licenses | |
| 13 | + end | |
| 14 | +end | ... | ... | 
| ... | ... | @@ -0,0 +1,11 @@ | 
| 1 | +class AddLicenseToArticle < ActiveRecord::Migration | |
| 2 | + def self.up | |
| 3 | + add_column :articles, :license_id, :integer | |
| 4 | + add_column :article_versions, :license_id, :integer | |
| 5 | + end | |
| 6 | + | |
| 7 | + def self.down | |
| 8 | + remove_column :articles, :license_id | |
| 9 | + remove_column :article_versions, :license_id | |
| 10 | + end | |
| 11 | +end | ... | ... | 
db/migrate/20120716161506_add_manage_environment_license_to_admin_role.rb
0 → 100644
| ... | ... | @@ -0,0 +1,17 @@ | 
| 1 | +class AddManageEnvironmentLicenseToAdminRole < ActiveRecord::Migration | |
| 2 | + def self.up | |
| 3 | + Environment.all.map(&:id).each do |id| | |
| 4 | + role = Environment::Roles.admin(id) | |
| 5 | + role.permissions << 'manage_environment_licenses' | |
| 6 | + role.save! | |
| 7 | + end | |
| 8 | + end | |
| 9 | + | |
| 10 | + def self.down | |
| 11 | + Environment.all.map(&:id).each do |id| | |
| 12 | + role = Environment::Roles.admin(id) | |
| 13 | + role.permissions -= ['manage_environment_licenses'] | |
| 14 | + role.save! | |
| 15 | + end | |
| 16 | + end | |
| 17 | +end | ... | ... | 
db/migrate/20120718145131_add_manage_environment_templates_to_admin_role.rb
0 → 100644
| ... | ... | @@ -0,0 +1,17 @@ | 
| 1 | +class AddManageEnvironmentTemplatesToAdminRole < ActiveRecord::Migration | |
| 2 | + def self.up | |
| 3 | + Environment.all.map(&:id).each do |id| | |
| 4 | + role = Environment::Roles.admin(id) | |
| 5 | + role.permissions << 'manage_environment_templates' | |
| 6 | + role.save! | |
| 7 | + end | |
| 8 | + end | |
| 9 | + | |
| 10 | + def self.down | |
| 11 | + Environment.all.map(&:id).each do |id| | |
| 12 | + role = Environment::Roles.admin(id) | |
| 13 | + role.permissions -= ['manage_environment_templates'] | |
| 14 | + role.save! | |
| 15 | + end | |
| 16 | + end | |
| 17 | +end | ... | ... | 
| ... | ... | @@ -0,0 +1,27 @@ | 
| 1 | +class CreateDefaultLicenses < ActiveRecord::Migration | |
| 2 | + def self.up | |
| 3 | + Environment.all.each do |environment| | |
| 4 | + License.create!(:name => 'CC (by)', :url => 'http://creativecommons.org/licenses/by/3.0/legalcode', :environment => environment) | |
| 5 | + License.create!(:name => 'CC (by-nd)', :url => 'http://creativecommons.org/licenses/by-nd/3.0/legalcode', :environment => environment) | |
| 6 | + License.create!(:name => 'CC (by-sa)', :url => 'http://creativecommons.org/licenses/by-sa/3.0/legalcode', :environment => environment) | |
| 7 | + License.create!(:name => 'CC (by-nc)', :url => 'http://creativecommons.org/licenses/by-nc/3.0/legalcode', :environment => environment) | |
| 8 | + License.create!(:name => 'CC (by-nc-nd)', :url => 'http://creativecommons.org/licenses/by-nc-nd/3.0/legalcode', :environment => environment) | |
| 9 | + License.create!(:name => 'CC (by-nc-sa)', :url => 'http://creativecommons.org/licenses/by-nc-sa/3.0/legalcode', :environment => environment) | |
| 10 | + License.create!(:name => 'Free Art', :url => 'http://artlibre.org/licence/lal/en', :environment => environment) | |
| 11 | + License.create!(:name => 'GNU FDL', :url => 'http://www.gnu.org/licenses/fdl-1.3.txt', :environment => environment) | |
| 12 | + end | |
| 13 | + end | |
| 14 | + | |
| 15 | + def self.down | |
| 16 | + licenses = [] | |
| 17 | + licenses += License.find(:all, :conditions => {:name => 'CC (by)'}) | |
| 18 | + licenses += License.find(:all, :conditions => {:name => 'CC (by-nd)'}) | |
| 19 | + licenses += License.find(:all, :conditions => {:name => 'CC (by-sa)'}) | |
| 20 | + licenses += License.find(:all, :conditions => {:name => 'CC (by-nc)'}) | |
| 21 | + licenses += License.find(:all, :conditions => {:name => 'CC (by-nc-nd)'}) | |
| 22 | + licenses += License.find(:all, :conditions => {:name => 'CC (by-nc-sa)'}) | |
| 23 | + licenses += License.find(:all, :conditions => {:name => 'Free Art'}) | |
| 24 | + licenses += License.find(:all, :conditions => {:name => 'GNU FDL'}) | |
| 25 | + licenses.compact.map(&:destroy) | |
| 26 | + end | |
| 27 | +end | ... | ... | 
db/migrate/20120813163139_set_activation_code_to_nil_if_already_activated.rb
0 → 100644
db/migrate/20120818030329_remove_action_tracker_with_target_nil.rb
0 → 100644
| ... | ... | @@ -0,0 +1,14 @@ | 
| 1 | +class RemoveActionTrackerWithTargetNil < ActiveRecord::Migration | |
| 2 | + def self.up | |
| 3 | + select_all("SELECT id FROM action_tracker").each do |tracker| | |
| 4 | + activity = ActionTracker::Record.find_by_id(tracker['id']) | |
| 5 | + if activity && activity.target.nil? | |
| 6 | + activity.destroy | |
| 7 | + end | |
| 8 | + end | |
| 9 | + end | |
| 10 | + | |
| 11 | + def self.down | |
| 12 | + say "this migration can't be reverted" | |
| 13 | + end | |
| 14 | +end | ... | ... | 
db/schema.rb
| ... | ... | @@ -9,7 +9,7 @@ | 
| 9 | 9 | # | 
| 10 | 10 | # It's strongly recommended to check this file into your version control system. | 
| 11 | 11 | |
| 12 | -ActiveRecord::Schema.define(:version => 20120411132751) do | |
| 12 | +ActiveRecord::Schema.define(:version => 20120818030329) do | |
| 13 | 13 | |
| 14 | 14 | create_table "abuse_reports", :force => true do |t| | 
| 15 | 15 | t.integer "reporter_id" | 
| ... | ... | @@ -85,6 +85,7 @@ ActiveRecord::Schema.define(:version => 20120411132751) do | 
| 85 | 85 | t.integer "translation_of_id" | 
| 86 | 86 | t.string "language" | 
| 87 | 87 | t.string "source_name" | 
| 88 | + t.integer "license_id" | |
| 88 | 89 | end | 
| 89 | 90 | |
| 90 | 91 | create_table "articles", :force => true do |t| | 
| ... | ... | @@ -125,6 +126,7 @@ ActiveRecord::Schema.define(:version => 20120411132751) do | 
| 125 | 126 | t.integer "translation_of_id" | 
| 126 | 127 | t.string "language" | 
| 127 | 128 | t.string "source_name" | 
| 129 | + t.integer "license_id" | |
| 128 | 130 | end | 
| 129 | 131 | |
| 130 | 132 | add_index "articles", ["translation_of_id"], :name => "index_articles_on_translation_of_id" | 
| ... | ... | @@ -314,6 +316,13 @@ ActiveRecord::Schema.define(:version => 20120411132751) do | 
| 314 | 316 | add_index "inputs", ["product_category_id"], :name => "index_inputs_on_product_category_id" | 
| 315 | 317 | add_index "inputs", ["product_id"], :name => "index_inputs_on_product_id" | 
| 316 | 318 | |
| 319 | + create_table "licenses", :force => true do |t| | |
| 320 | + t.string "name", :null => false | |
| 321 | + t.string "slug", :null => false | |
| 322 | + t.string "url" | |
| 323 | + t.integer "environment_id", :null => false | |
| 324 | + end | |
| 325 | + | |
| 317 | 326 | create_table "mailing_sents", :force => true do |t| | 
| 318 | 327 | t.integer "mailing_id" | 
| 319 | 328 | t.integer "person_id" | 
| ... | ... | @@ -425,6 +434,8 @@ ActiveRecord::Schema.define(:version => 20120411132751) do | 
| 425 | 434 | t.boolean "validated", :default => true | 
| 426 | 435 | t.string "cnpj" | 
| 427 | 436 | t.string "national_region_code" | 
| 437 | + t.boolean "is_template", :default => false | |
| 438 | + t.integer "template_id" | |
| 428 | 439 | end | 
| 429 | 440 | |
| 430 | 441 | add_index "profiles", ["environment_id"], :name => "index_profiles_on_environment_id" | ... | ... | 
debian/changelog
| 1 | +noosfero (0.38.1) unstable; urgency=low | |
| 2 | + | |
| 3 | + * Bugfixes release | |
| 4 | + | |
| 5 | + -- Daniela Soares Feitosa <daniela@colivre.coop.br> Sat, 18 Aug 2012 07:49:59 -0300 | |
| 6 | + | |
| 7 | +noosfero (0.38.0) unstable; urgency=low | |
| 8 | + | |
| 9 | + * Features version release | |
| 10 | + | |
| 11 | + -- Daniela Soares Feitosa <daniela@colivre.coop.br> Tue, 31 Jul 2012 19:51:43 -0300 | |
| 12 | + | |
| 1 | 13 | noosfero (0.37.0) unstable; urgency=low | 
| 2 | 14 | |
| 3 | 15 | * Features version release | ... | ... | 
debian/control
| ... | ... | @@ -47,11 +47,12 @@ Depends: | 
| 47 | 47 | memcached, | 
| 48 | 48 | debconf, | 
| 49 | 49 | dbconfig-common, | 
| 50 | - postgresql, | |
| 51 | 50 | adduser, | 
| 52 | 51 | exim4 | mail-transport-agent, | 
| 53 | 52 | ${misc:Depends} | 
| 54 | -Recommends: postgresql-client | |
| 53 | +Recommends: | |
| 54 | + postgresql, | |
| 55 | + postgresql-client | |
| 55 | 56 | Description: free web-based platform for social networks | 
| 56 | 57 | Noosfero is a web platform for social and solidarity economy networks with | 
| 57 | 58 | blog, e-Porfolios, CMS, RSS, thematic discussion, events agenda and collective | ... | ... | 
debian/thin.yml
etc/logrotate.d/noosfero
features/edit_environment_templates.feature
| 1 | 1 | Feature: edit environment templates | 
| 2 | 2 | As an administrator | 
| 3 | - I want edit templates | |
| 3 | + I want to edit templates | |
| 4 | 4 | |
| 5 | 5 | Background: | 
| 6 | 6 | Given that the default environment have all profile templates | 
| ... | ... | @@ -9,37 +9,37 @@ Feature: edit environment templates | 
| 9 | 9 | Given I am logged in as admin | 
| 10 | 10 | When I follow "Administration" | 
| 11 | 11 | And I follow "Edit Templates" | 
| 12 | - Then I should see "Edit Person Template" link | |
| 13 | - And I should see "Edit Community Template" link | |
| 14 | - And I should see "Edit Enterprise Template" link | |
| 15 | - And I should see "Edit Inactive Enterprise Template" link | |
| 12 | + Then I should see "Person template" link | |
| 13 | + And I should see "Community template" link | |
| 14 | + And I should see "Enterprise template" link | |
| 15 | + And I should see "Inactive Enterprise template" link | |
| 16 | 16 | |
| 17 | 17 | Scenario: Go to control panel of person template | 
| 18 | 18 | Given I am logged in as admin | 
| 19 | 19 | When I follow "Administration" | 
| 20 | 20 | And I follow "Edit Templates" | 
| 21 | - And I follow "Edit Person Template" | |
| 21 | + And I follow "Person template" | |
| 22 | 22 | Then I should be on Person template's control panel | 
| 23 | 23 | |
| 24 | 24 | Scenario: Go to control panel of enterprise template | 
| 25 | 25 | Given I am logged in as admin | 
| 26 | 26 | When I follow "Administration" | 
| 27 | 27 | And I follow "Edit Templates" | 
| 28 | - And I follow "Edit Enterprise Template" | |
| 28 | + And I follow "Enterprise template" | |
| 29 | 29 | Then I should be on Enterprise template's control panel | 
| 30 | 30 | |
| 31 | 31 | Scenario: Go to control panel of inactive enterprise template | 
| 32 | 32 | Given I am logged in as admin | 
| 33 | 33 | When I follow "Administration" | 
| 34 | 34 | And I follow "Edit Templates" | 
| 35 | - And I follow "Edit Inactive Enterprise Template" | |
| 35 | + And I follow "Inactive enterprise template" | |
| 36 | 36 | Then I should be on Inactive Enterprise template's control panel | 
| 37 | 37 | |
| 38 | 38 | Scenario: Go to control panel of community template | 
| 39 | 39 | Given I am logged in as admin | 
| 40 | 40 | When I follow "Administration" | 
| 41 | 41 | And I follow "Edit Templates" | 
| 42 | - And I follow "Edit Community Template" | |
| 42 | + And I follow "Community template" | |
| 43 | 43 | Then I should be on Community template's control panel | 
| 44 | 44 | |
| 45 | 45 | Scenario: Not see link to edit an unexistent template | 
| ... | ... | @@ -47,7 +47,7 @@ Feature: edit environment templates | 
| 47 | 47 | And I am logged in as admin | 
| 48 | 48 | When I follow "Administration" | 
| 49 | 49 | And I follow "Edit Templates" | 
| 50 | - Then I should see "Edit Person Template" link | |
| 51 | - And I should see "Edit Community Template" link | |
| 52 | - And I should see "Edit Enterprise Template" link | |
| 53 | - And I should not see "Edit Inactive Enterprise Template" link | |
| 50 | + Then I should see "Person template" link | |
| 51 | + And I should see "Community template" link | |
| 52 | + And I should see "Enterprise template" link | |
| 53 | + And I should not see "Inactive enterprise template" link | ... | ... | 
features/private_profile.feature
| ... | ... | @@ -30,16 +30,6 @@ Feature: private profiles | 
| 30 | 30 | When I go to shygirl's homepage | 
| 31 | 31 | Then I should not see "Add friend" | 
| 32 | 32 | |
| 33 | - Scenario: viewing a private community profile shouldn't show the news if not logged or not a member | |
| 34 | - Given I am on Safernet's homepage | |
| 35 | - Then I should not see "What's new" | |
| 36 | - And I am logged in as "joao" | |
| 37 | - When I am on Safernet's homepage | |
| 38 | - Then I should not see "What's new" | |
| 39 | - And "joao" is a member of "Safernet" | |
| 40 | - When I am on Safernet's homepage | |
| 41 | - Then I should see "What's new" | |
| 42 | - | |
| 43 | 33 | Scenario: person private profiles should not display sensible information | 
| 44 | 34 | Given I am logged in as "joao" | 
| 45 | 35 | When I go to shygirl's homepage | ... | ... | 
| ... | ... | @@ -0,0 +1,14 @@ | 
| 1 | +Then /^I directly delete content with name "([^\"]*)" for testing purposes$/ do |content_name| | |
| 2 | + Article.find_by_name(content_name).destroy | |
| 3 | +end | |
| 4 | + | |
| 5 | +Then /^I should be at the url "([^\"]*)"$/ do |url| | |
| 6 | + if response.class.to_s == 'Webrat::SeleniumResponse' | |
| 7 | + URI.parse(response.selenium.get_location).path.should == url | |
| 8 | + else | |
| 9 | + URI.parse(current_url).path.should == url | |
| 10 | + end | |
| 11 | +end | |
| 12 | + | |
| 13 | +Then /^I don't fill anything$/ do | |
| 14 | +end | ... | ... | 
lib/noosfero.rb
plugins/mezuro/AUTHORS
| 1 | -KALIBRO PLUGIN AUTHORS | |
| 2 | -===================== | |
| 1 | +Mezuro Authors | |
| 2 | +============== | |
| 3 | 3 | |
| 4 | -Copyright 2010-2011 | |
| 5 | - Almir Alves Pereira (almir.sne at gmail.com) | |
| 6 | - Andre Casimiro (ah.casimiro at gmail.com) | |
| 7 | - Carlos Morais (carlos88morais at gmail.com) | |
| 8 | - Everton Santos (everton2x4 at gmail.com) | |
| 9 | - Paulo Meirelles (paulo at softwarelivre.org) | |
| 10 | - Rafael Messias (rmmartins at gmail.com) | |
| 4 | +Copyright 2010-2012 | |
| 5 | +------------------- | |
| 11 | 6 | |
| 12 | -Collaborators: | |
| 13 | -- Ana Paula Oliveira dos Santos (ana at ime.usp.br) | |
| 14 | -- Lucianna Almeida (lucianna.th at gmail.com) | |
| 15 | -- Joao Machini (joao.machini at gmail.com) | |
| 16 | -- Rodrigo Souto (rodrigo@colivre.coop.br) | |
| 17 | -- Thiago Colucci (ticolucci at gmail.com) | |
| 18 | -- Vinicius Daros (vinicius.k.daros at gmail.com) | |
| 19 | -- Viviane Almeida Santos (viviane.almeida at gmail.com) | |
| 7 | + Almir Alves Pereira (almir.sne at gmail.com) | |
| 8 | + Alessandro Palmeira (alessandro.palmeira at gmail.com) | |
| 9 | + Andre Casimiro (ah.casimiro at gmail.com) | |
| 10 | + Antonio Terceiro (terceiro at colivre.coop.br) | |
| 11 | + Caio Salgado (caio.csalgado at gmail.com) | |
| 12 | + Carlos Morais (carlos88morais at gmail.com) | |
| 13 | + Diego Araújo (diegoamc90 at gmail.com) | |
| 14 | + Everton Santos (everton2x4 at gmail.com) | |
| 15 | + Jefferson Fernandes (jeffs.fernandes at gmail.com) | |
| 16 | + Joao Machini (joao.machini at gmail.com) | |
| 17 | + João da Silva (jaodsilv@linux.ime.usp.br) | |
| 18 | + Paulo Meirelles (paulo at softwarelivre.org) | |
| 19 | + Rafael Manso (rr.manzo at gmail.com) | |
| 20 | + Rafael Messias (rmmartins at gmail.com) | |
| 21 | + Renan Teruo (renanteruoc at gmail.com) | |
| 22 | + Rodrigo Souto (rodrigo at colivre.coop.br) | |
| 23 | + | |
| 24 | +Collaborators (from USP Lab XP course 2010 on another code) | |
| 25 | +----------------------------------------------------------- | |
| 26 | + | |
| 27 | + Ana Paula Oliveira dos Santos (anapaulao.santos at gmail.com) | |
| 28 | + Lucianna Almeida (lucianna.th at gmail.com) | |
| 29 | + Thiago Colucci (ticolucci at gmail.com) | |
| 30 | + Vinicius Daros (vinicius.k.daros at gmail.com) | |
| 31 | + Viviane Almeida Santos (viviane.almeida at gmail.com) | |
| 32 | + | |
| 33 | +Advisors | |
| 34 | +-------- | |
| 35 | + | |
| 36 | + Fabio Kon (fabio.kon at ime.usp.br) | |
| 37 | + Alfredo Goldman (gold at ime.usp.br) | |
| 20 | 38 | ... | ... | 
plugins/mezuro/README
| 1 | -README - Mezuro (Kalibro Plugin) | |
| 1 | +README - Mezuro (Mezuro Plugin) | |
| 2 | 2 | ================================ | 
| 3 | 3 | |
| 4 | -Mezuro is source code tracking network based on Noosfero Platform with Kalibro Plugin to access Kalibro Web Service. | |
| 4 | +Mezuro is a source code tracking platform based on Noosfero social networking | |
| 5 | +platform with Mezuro Plugin actived to access Kalibro Web Service. | |
| 6 | + | |
| 5 | 7 | |
| 6 | 8 | INSTALL | 
| 7 | 9 | ======= | 
| 8 | 10 | |
| 9 | -See the Noosfero install file and include Ruby gem Savon: | |
| 11 | +Dependences | |
| 12 | +----------- | |
| 13 | + | |
| 14 | +See the Noosfero install file. After install Noosfero, install Mezuro dependences: | |
| 10 | 15 | |
| 11 | -gem install savon | |
| 16 | +$ gem install nokogiri -v 1.5.0 | |
| 17 | +$ gem install savon -v 0.9.7 | |
| 18 | +$ gem install googlecharts | |
| 12 | 19 | |
| 13 | -Also, you need to enable Mezuro Plugin in the Noosfero: | |
| 20 | +Enable Plugin | |
| 21 | +------------- | |
| 22 | + | |
| 23 | +Also, you need to enable Mezuro Plugin at you Noosfero: | |
| 14 | 24 | |
| 15 | 25 | cd <your_noosfero_dir> | 
| 16 | 26 | ./script/noosfero-plugins enable mezuro | 
| 17 | 27 | |
| 18 | -To run Mezuro (kalibro Plugin) and its acceptance tests, you need to install the Kalibro Service. | |
| 28 | + | |
| 29 | +Install Service | |
| 30 | +--------------- | |
| 31 | + | |
| 32 | +To run Mezuro (Noosfero with Mezuro Plugin), you need to install the Kalibro | |
| 33 | +Service. | |
| 34 | + | |
| 19 | 35 | For that, access https://gitorious.org/kalibro/kalibro/blobs/master/INSTALL | 
| 20 | 36 | |
| 21 | 37 | |
| 22 | -SOURCE CODE | |
| 38 | +Configure Service Address | |
| 39 | +------------------------- | |
| 40 | + | |
| 41 | +Finally, copy service.yaml.example to service.yaml and define your Kalibro | |
| 42 | +Service address: | |
| 43 | + | |
| 44 | +$ cd <your_noosfero_dir>/plugin/mezuro | |
| 45 | +$ cp service.yaml.example service.yaml | |
| 46 | + | |
| 47 | +If you install Kalibro Service at localhost, just keep the default | |
| 48 | +adress: | |
| 49 | + | |
| 50 | +http://localhost:8080/KalibroService/ | |
| 51 | + | |
| 52 | +Applying Mezuro Theme | |
| 53 | +--------------------- | |
| 54 | + | |
| 55 | +$ cd public/designs/themes && rm -f default | |
| 56 | +$ git clone git://gitorious.org/mezuro/mezuro-theme.git | |
| 57 | +$ ln -s mezuro-theme/ default && cd ../../../ | |
| 58 | + | |
| 59 | +Active Plugin | |
| 60 | +------------- | |
| 61 | + | |
| 62 | +As a Noosfero administrator user, go to administrator panel: | |
| 63 | + | |
| 64 | +- Click on "Enable/disable plugins" option | |
| 65 | +- Click on "Mezuro Plugin" check-box | |
| 66 | + | |
| 67 | + | |
| 68 | +DEVELOPMENT | |
| 23 | 69 | =========== | 
| 24 | 70 | |
| 25 | -To get the Mezuro (Noosfero with Kalibro Plugin) development repository: | |
| 71 | +Get the Mezuro (Noosfero with Mezuro Plugin) development repository: | |
| 26 | 72 | |
| 27 | -git clone https://gitorious.org/+mezuro/noosfero/mezuro-noosfero-plugin | |
| 28 | -cd mezuro-noosfero-plugin | |
| 29 | -git checkout mezuro | |
| 73 | +$ git clone https://gitorious.org/+mezuro/noosfero/mezuro | |
| 74 | +$ cd mezuro | |
| 75 | +$ git checkout mezuro | |
| 30 | 76 | |
| 31 | -AUTHORS | |
| 32 | -======= | |
| 77 | +Running Mezuro tests | |
| 78 | +-------------------- | |
| 33 | 79 | |
| 34 | -Please see the file AUTHORS. | |
| 80 | +$ rake test:noosfero_plugins:mezuro | |
| 35 | 81 | |
| 36 | -BUGS | |
| 37 | -==== | |
| 38 | 82 | |
| 39 | -If you found any bug, please report at mezuro@listas.softwarelivre.org | |
| 40 | -(http://listas.softwarelivre.org/cgi-bin/mailman/listinfo/mezuro) | |
| 83 | +Get Involved | |
| 84 | +============ | |
| 85 | + | |
| 86 | +If you found any bug and/or want to collaborate, please send an e-mail to | |
| 87 | +paulo@softwarelivre.org | |
| 88 | + | |
| 41 | 89 | |
| 42 | 90 | LICENSE | 
| 43 | 91 | ======= | 
| 44 | -Copyright (c) The Authors developers. | |
| 92 | + | |
| 93 | +Copyright (c) The Author developers. | |
| 45 | 94 | |
| 46 | 95 | See Noosfero license. | 
| 47 | 96 | |
| 48 | 97 | |
| 98 | +AUTHORS | |
| 99 | +======= | |
| 100 | + | |
| 101 | +Please, see the Mezuro AUTHORS file. | |
| 102 | + | |
| 103 | + | |
| 49 | 104 | ACKNOWLEDGMENTS | 
| 50 | 105 | =============== | 
| 51 | 106 | |
| 52 | 107 | The authors have been supported by organizations: | 
| 53 | 108 | |
| 109 | +University of São Paulo (USP) | |
| 110 | +FLOSS Competence Center | |
| 111 | +http://ccsl.ime.usp.br | |
| 112 | + | |
| 54 | 113 | Brazilian National Research Council (CNPQ) | 
| 55 | 114 | http://www.cnpq.br/ | 
| 56 | - | |
| 57 | -USP FLOSS Competence Center | |
| 58 | -http://ccsl.ime.usp.br/ | ... | ... | 
| ... | ... | @@ -0,0 +1,73 @@ | 
| 1 | +README/TODO do branch cucumber_tests | |
| 2 | + | |
| 3 | +Tarefas: | |
| 4 | +- Escrever uma história (procurar uma já escrita) para isso | |
| 5 | +- Descobrir o porquê dos erros nos testes do mezuro (repository_url) | |
| 6 | +- Dar rebase com o mezuro-dev (os tais testes passam no mezuro-dev) | |
| 7 | +- Fazer mais testes | |
| 8 | +- Ver como/quando o selenium pode ser integrado ao projeto (conversar com noosfero/ talvez até tentar implementar alguma coisa??) | |
| 9 | + | |
| 10 | + | |
| 11 | +Testes de aceitação a serem feitos: (* já feito) | |
| 12 | + | |
| 13 | +Projetos: | |
| 14 | + Criar | |
| 15 | + * correto | |
| 16 | + * errado | |
| 17 | + duplicado | |
| 18 | + Editar | |
| 19 | + correto | |
| 20 | + errado | |
| 21 | + Deletar(não precisa fazer - problema do noosfero) | |
| 22 | + | |
| 23 | +Configurações: | |
| 24 | + criar | |
| 25 | + *correta | |
| 26 | + *errado (sem titulo) | |
| 27 | + *duplicada | |
| 28 | + editar - verificar se mantem as metricas | |
| 29 | + correto | |
| 30 | + não é possível mudar o titulo | |
| 31 | + deletar (não precisa fazer - problema do noosfero) | |
| 32 | + metricas: | |
| 33 | + criar | |
| 34 | + nativa: | |
| 35 | + *correta | |
| 36 | + *errada | |
| 37 | + duplicada (é pra funcionar?) | |
| 38 | + composta: | |
| 39 | + *correta | |
| 40 | + errada | |
| 41 | + duplicada | |
| 42 | + editar | |
| 43 | + para metrica correta | |
| 44 | + para metrica errada | |
| 45 | + *deletar | |
| 46 | + ranges: | |
| 47 | + criar | |
| 48 | + *range correto | |
| 49 | + range errado | |
| 50 | + todos os casos | |
| 51 | + editar | |
| 52 | + para range correto | |
| 53 | + para range errado | |
| 54 | + deletar | |
| 55 | + | |
| 56 | +Testes falhando: | |
| 57 | + | |
| 58 | +Arquivo adding_metric_configuration.feature: | |
| 59 | + Scenario: adding a native metric configuration without code | |
| 60 | + Precisa do selenium para ver em qual página está. | |
| 61 | + Scenario: adding a compound metric configuration | |
| 62 | + Scenario: adding a compound metric configuration with invalid script | |
| 63 | + As métricas compostas não estão salvando direito. | |
| 64 | + | |
| 65 | + | |
| 66 | +Arquivo creating_project.feature: | |
| 67 | + Scenario: I create a Kalibro project with valid attributes | |
| 68 | + ás vezes falha por erro de já existir na Kalibro. Esse erro teóricamente já havia sido resolvido. | |
| 69 | + | |
| 70 | + | |
| 71 | +Arquivo adding_ranges.feature: | |
| 72 | + Scenario: adding a range to a metric configuration | |
| 73 | + Precisa do selenium para esperar a página carregar. | ... | ... | 
plugins/mezuro/controllers/mezuro_plugin_myprofile_controller.rb
0 → 100644
| ... | ... | @@ -0,0 +1,126 @@ | 
| 1 | +class MezuroPluginMyprofileController < ProfileController | |
| 2 | + | |
| 3 | + append_view_path File.join(File.dirname(__FILE__) + '/../views') | |
| 4 | + | |
| 5 | + | |
| 6 | + def choose_base_tool | |
| 7 | + @configuration_content = profile.articles.find(params[:id]) | |
| 8 | + @base_tools = Kalibro::BaseTool.all_names | |
| 9 | + end | |
| 10 | + | |
| 11 | + def choose_metric | |
| 12 | + @configuration_content = profile.articles.find(params[:id]) | |
| 13 | + @base_tool = params[:base_tool] | |
| 14 | + @supported_metrics = Kalibro::BaseTool.find_by_name(@base_tool).supported_metrics | |
| 15 | + end | |
| 16 | + | |
| 17 | + def new_metric_configuration | |
| 18 | + @configuration_content = profile.articles.find(params[:id]) | |
| 19 | + @metric = Kalibro::BaseTool.find_by_name(params[:base_tool]).metric params[:metric_name] | |
| 20 | + end | |
| 21 | + | |
| 22 | + def new_compound_metric_configuration | |
| 23 | + @configuration_content = profile.articles.find(params[:id]) | |
| 24 | + @metric_configurations = @configuration_content.metric_configurations | |
| 25 | + end | |
| 26 | + | |
| 27 | + def edit_metric_configuration | |
| 28 | + @configuration_content = profile.articles.find(params[:id]) | |
| 29 | + @metric_configuration = Kalibro::MetricConfiguration.find_by_configuration_name_and_metric_name(@configuration_content.name, params[:metric_name]) | |
| 30 | + @metric = @metric_configuration.metric | |
| 31 | + end | |
| 32 | + | |
| 33 | + def edit_compound_metric_configuration | |
| 34 | + @configuration_content = profile.articles.find(params[:id]) | |
| 35 | + @metric_configuration = Kalibro::MetricConfiguration.find_by_configuration_name_and_metric_name(@configuration_content.name, params[:metric_name]) | |
| 36 | + @metric_configurations = @configuration_content.metric_configurations | |
| 37 | + @metric = @metric_configuration.metric | |
| 38 | + end | |
| 39 | + | |
| 40 | + def create_metric_configuration | |
| 41 | + id = params[:id] | |
| 42 | + metric_name = params[:metric_configuration][:metric][:name] | |
| 43 | + (Kalibro::MetricConfiguration.new(params[:metric_configuration])).save | |
| 44 | + redirect_to "/myprofile/#{profile.identifier}/plugin/mezuro/edit_metric_configuration?id=#{id}&metric_name=#{metric_name.gsub(/\s/, '+')}" | |
| 45 | + end | |
| 46 | + | |
| 47 | + def create_compound_metric_configuration | |
| 48 | + id = params[:id] | |
| 49 | + metric_name = params[:metric_configuration][:metric][:name] | |
| 50 | + Kalibro::MetricConfiguration.new(params[:metric_configuration]).save | |
| 51 | + redirect_to "/myprofile/#{profile.identifier}/plugin/mezuro/edit_compound_metric_configuration?id=#{id}&metric_name=#{metric_name.gsub(/\s/, '+')}" | |
| 52 | + end | |
| 53 | + | |
| 54 | + def update_metric_configuration | |
| 55 | + @configuration_content = profile.articles.find(params[:id]) | |
| 56 | + metric_name = params[:metric_configuration][:metric][:name] | |
| 57 | + metric_configuration = Kalibro::MetricConfiguration.find_by_configuration_name_and_metric_name(@configuration_content.name, metric_name) | |
| 58 | + metric_configuration.update_attributes params[:metric_configuration] | |
| 59 | + redirect_to "/#{profile.identifier}/#{@configuration_content.slug}" | |
| 60 | + end | |
| 61 | + | |
| 62 | + def update_compound_metric_configuration | |
| 63 | + @configuration_content = profile.articles.find(params[:id]) | |
| 64 | + metric_name = params[:metric_configuration][:metric][:name] | |
| 65 | + metric_configuration = Kalibro::MetricConfiguration.find_by_configuration_name_and_metric_name(@configuration_content.name, metric_name) | |
| 66 | + metric_configuration.update_attributes params[:metric_configuration] | |
| 67 | + redirect_to "/#{profile.identifier}/#{@configuration_content.slug}" | |
| 68 | + end | |
| 69 | + | |
| 70 | + def remove_metric_configuration | |
| 71 | + configuration_content = profile.articles.find(params[:id]) | |
| 72 | + metric_name = params[:metric_name] | |
| 73 | + metric_configuration = Kalibro::MetricConfiguration.find_by_configuration_name_and_metric_name(configuration_content.name, metric_name) | |
| 74 | + metric_configuration.destroy | |
| 75 | + redirect_to "/#{profile.identifier}/#{configuration_content.slug}" | |
| 76 | + end | |
| 77 | + | |
| 78 | + def new_range | |
| 79 | + @configuration_content = profile.articles.find(params[:id]) | |
| 80 | + @metric_name = params[:metric_name] | |
| 81 | + @range = Kalibro::Range.new | |
| 82 | + end | |
| 83 | + | |
| 84 | + def edit_range | |
| 85 | + @configuration_content = profile.articles.find(params[:id]) | |
| 86 | + @metric_name = params[:metric_name] | |
| 87 | + @beginning_id = params[:beginning_id] | |
| 88 | + metric_configuration = Kalibro::MetricConfiguration.find_by_configuration_name_and_metric_name(@configuration_content.name, @metric_name) | |
| 89 | + @range = metric_configuration.ranges.find{|range| range.beginning == @beginning_id.to_f || @beginning_id =="-INF" } | |
| 90 | + end | |
| 91 | + | |
| 92 | + def create_range | |
| 93 | + @configuration_content = profile.articles.find(params[:id]) | |
| 94 | + @range = Kalibro::Range.new params[:range] | |
| 95 | + metric_name = params[:metric_name] | |
| 96 | + metric_configuration = Kalibro::MetricConfiguration.find_by_configuration_name_and_metric_name(@configuration_content.name, metric_name) | |
| 97 | + metric_configuration.add_range(@range) | |
| 98 | + metric_configuration.save | |
| 99 | + end | |
| 100 | + | |
| 101 | + def update_range | |
| 102 | + configuration_content = profile.articles.find(params[:id]) | |
| 103 | + metric_name = params[:metric_name] | |
| 104 | + beginning_id = params[:beginning_id] | |
| 105 | + metric_configuration = Kalibro::MetricConfiguration.find_by_configuration_name_and_metric_name(configuration_content.name, metric_name) | |
| 106 | + index = metric_configuration.ranges.index{ |range| range.beginning == beginning_id.to_f || beginning_id == "-INF" } | |
| 107 | + metric_configuration.ranges[index] = Kalibro::Range.new params[:range] | |
| 108 | + metric_configuration.save | |
| 109 | + end | |
| 110 | + | |
| 111 | + def remove_range | |
| 112 | + configuration_content = profile.articles.find(params[:id]) | |
| 113 | + metric_name = params[:metric_name] | |
| 114 | + beginning_id = params[:beginning_id] | |
| 115 | + metric_configuration = Kalibro::MetricConfiguration.find_by_configuration_name_and_metric_name(configuration_content.name, metric_name) | |
| 116 | + metric_configuration.ranges.delete_if { |range| range.beginning == beginning_id.to_f || beginning_id == "-INF" } | |
| 117 | + metric_configuration.save | |
| 118 | + formatted_metric_name = metric_name.gsub(/\s/, '+') | |
| 119 | + if metric_configuration.metric.class == Kalibro::CompoundMetric | |
| 120 | + redirect_to "/myprofile/#{profile.identifier}/plugin/mezuro/edit_compound_metric_configuration?id=#{configuration_content.id}&metric_name=#{formatted_metric_name}" | |
| 121 | + else | |
| 122 | + redirect_to "/myprofile/#{profile.identifier}/plugin/mezuro/edit_metric_configuration?id=#{configuration_content.id}&metric_name=#{formatted_metric_name}" | |
| 123 | + end | |
| 124 | + end | |
| 125 | + | |
| 126 | +end | ... | ... | 
plugins/mezuro/controllers/mezuro_plugin_profile_controller.rb
| 1 | 1 | class MezuroPluginProfileController < ProfileController | 
| 2 | 2 | |
| 3 | - def metrics | |
| 4 | - project_content = profile.articles.find(params[:id]) | |
| 5 | - module_name = params[:module_name] | |
| 6 | - render :partial => 'content_viewer/module_result', :locals => { :module_result => project_content.module_result(module_name) } | |
| 3 | + append_view_path File.join(File.dirname(__FILE__) + '/../views') | |
| 4 | + | |
| 5 | + def project_state | |
| 6 | + @content = profile.articles.find(params[:id]) | |
| 7 | + project = @content.project | |
| 8 | + state = project.error.nil? ? project.state : "ERROR" | |
| 9 | + render :text => state | |
| 7 | 10 | end | 
| 8 | 11 | |
| 12 | + def project_error | |
| 13 | + @content = profile.articles.find(params[:id]) | |
| 14 | + @project = @content.project | |
| 15 | + render :partial => 'content_viewer/project_error' | |
| 16 | + end | |
| 17 | + | |
| 18 | + def project_result | |
| 19 | + @content = profile.articles.find(params[:id]) | |
| 20 | + date = params[:date] | |
| 21 | + @project_result = date.nil? ? @content.project_result : @content.project_result_with_date(date) | |
| 22 | + render :partial => 'content_viewer/project_result' | |
| 23 | + end | |
| 24 | + | |
| 25 | + def module_result | |
| 26 | + @content = profile.articles.find(params[:id]) | |
| 27 | + @module_result = @content.module_result(params) | |
| 28 | + render :partial => 'content_viewer/module_result' | |
| 29 | + end | |
| 30 | + | |
| 31 | + def project_tree | |
| 32 | + @content = profile.articles.find(params[:id]) | |
| 33 | + date = params[:date] | |
| 34 | + project_result = date.nil? ? @content.project_result : @content.project_result_with_date(date) | |
| 35 | + @project_name = @content.project.name | |
| 36 | + @source_tree = project_result.node_of(params[:module_name]) | |
| 37 | + render :partial =>'content_viewer/source_tree' | |
| 38 | + end | |
| 39 | + | |
| 40 | + def module_metrics_history | |
| 41 | + metric_name = params[:metric_name] | |
| 42 | + @content = profile.articles.find(params[:id]) | |
| 43 | + module_history = @content.result_history(params[:module_name]) | |
| 44 | + @score_history = filtering_metric_history(metric_name, module_history) | |
| 45 | + render :partial => 'content_viewer/score_history' | |
| 46 | + end | |
| 47 | + | |
| 48 | + def module_grade_history | |
| 49 | + @content = profile.articles.find(params[:id]) | |
| 50 | + modules_results = @content.result_history(params[:module_name]) | |
| 51 | + @score_history = modules_results.collect { |module_result| module_result.grade } | |
| 52 | + render :partial => 'content_viewer/score_history' | |
| 53 | + end | |
| 54 | + | |
| 55 | + private | |
| 56 | + | |
| 57 | + def filtering_metric_history(metric_name, module_history) | |
| 58 | + metrics_history = module_history.map do |module_result| | |
| 59 | + module_result.metric_results | |
| 60 | + end | |
| 61 | + metric_history = metrics_history.map do |array_of_metric_result| | |
| 62 | + (array_of_metric_result.select do |metric_result| | |
| 63 | + metric_result.metric.name.delete("() ") == metric_name | |
| 64 | + end).first | |
| 65 | + end | |
| 66 | + metric_history.map do |metric_result| | |
| 67 | + metric_result.value | |
| 68 | + end | |
| 69 | + end | |
| 9 | 70 | end | ... | ... | 
plugins/mezuro/features/mezuro.feature
| ... | ... | @@ -1,69 +0,0 @@ | 
| 1 | -Feature: mezuro content | |
| 2 | - As a noosfero user | |
| 3 | - I want to create a Kalibro project | |
| 4 | - | |
| 5 | - Background: | |
| 6 | - Given the following users | |
| 7 | - | login | name | | |
| 8 | - | joaosilva | Joao Silva | | |
| 9 | - And I am logged in as "joaosilva" | |
| 10 | - And "Mezuro" plugin is enabled | |
| 11 | - And the following community | |
| 12 | - | identifier | name | | |
| 13 | - | mycommunity | My Community | | |
| 14 | - And "Joao Silva" is admin of "My Community" | |
| 15 | - | |
| 16 | - Scenario: I see Kalibro project as an option to new content | |
| 17 | - Given I am on My Community's cms | |
| 18 | - When I follow "New content" | |
| 19 | - Then I should see "Kalibro project" | |
| 20 | - | |
| 21 | - Scenario: I see Kalibro project's input form | |
| 22 | - Given I am on My Community's cms | |
| 23 | - When I follow "New content" | |
| 24 | - And I follow "Kalibro project" | |
| 25 | - Then I should see "Title" | |
| 26 | - And I should see "License" | |
| 27 | - And I should see "Repository type" | |
| 28 | - And I should see "GIT" | |
| 29 | - And I should see "REMOTE_ZIP" | |
| 30 | - And I should see "REMOTE_TARBALL" | |
| 31 | - And I should see "SUBVERSION" | |
| 32 | - And I should see "Repository url" | |
| 33 | - And I should see "Configuration" | |
| 34 | - And I should see "Kalibro for Java" | |
| 35 | - | |
| 36 | - Scenario: I create a sample mezuro content | |
| 37 | - Given I am on My Community's cms | |
| 38 | - When I create a content of type "Kalibro project" with the following data | |
| 39 | - | Title | Sample project | | |
| 40 | - | License | BSD | | |
| 41 | - | Repository type | GIT | | |
| 42 | - | Repository url | git://example | | |
| 43 | - Then I should see "Sample project" | |
| 44 | - And I should see "Viewed one time" | |
| 45 | - And I should see "BSD" | |
| 46 | - | |
| 47 | - Scenario: I create a real mezuro content | |
| 48 | - Given I am on My Community's cms | |
| 49 | - When I create a content of type "Kalibro project" with the following data | |
| 50 | - | Title | Qt-Calculator | | |
| 51 | - | License | GPL 2.0 | | |
| 52 | - | Repository type | SUBVERSION | | |
| 53 | - | Repository url | https://qt-calculator.svn.sourceforge.net/svnroot/qt-calculator | | |
| 54 | - Then I should see "Qt-Calculator" | |
| 55 | - | |
| 56 | - Scenario: I see results from a real Kalibro project | |
| 57 | - Given I am on My Community's cms | |
| 58 | - When I create a content of type "Kalibro project" with the following data | |
| 59 | - | Title | Qt-Calculator | | |
| 60 | - | License | GPL | | |
| 61 | - | Repository type | SUBVERSION | | |
| 62 | - | Repository url | https://qt-calculator.svn.sourceforge.net/svnroot/qt-calculator | | |
| 63 | - | Configuration | Kalibro for Java | | |
| 64 | - Then I should see "Qt-Calculator" | |
| 65 | - And I should see "GPL" | |
| 66 | - And I should see "SUBVERSION" | |
| 67 | - And I should see "https://qt-calculator.svn.sourceforge.net/svnroot/qt-calculator" | |
| 68 | - And I should see "Kalibro for Java" | |
| 69 | - And I should see "Kalibro Service is loading the source code" | 
| ... | ... | @@ -0,0 +1,29 @@ | 
| 1 | +class Kalibro::BaseTool < Kalibro::Model | |
| 2 | + | |
| 3 | + attr_accessor :name, :description, :supported_metric | |
| 4 | + | |
| 5 | + def self.all_names | |
| 6 | + request("BaseTool", :get_base_tool_names)[:base_tool_name].to_a | |
| 7 | + end | |
| 8 | + | |
| 9 | + def self.find_by_name(base_tool_name) | |
| 10 | + new request("BaseTool", :get_base_tool, {:base_tool_name => base_tool_name})[:base_tool] | |
| 11 | + end | |
| 12 | + | |
| 13 | + def supported_metric=(value) | |
| 14 | + @supported_metric = Kalibro::NativeMetric.to_objects_array value | |
| 15 | + end | |
| 16 | + | |
| 17 | + def supported_metrics | |
| 18 | + @supported_metric | |
| 19 | + end | |
| 20 | + | |
| 21 | + def supported_metrics=(supported_metrics) | |
| 22 | + @supported_metric = supported_metrics | |
| 23 | + end | |
| 24 | + | |
| 25 | + def metric(name) | |
| 26 | + supported_metrics.find {|metric| metric.name == name} | |
| 27 | + end | |
| 28 | + | |
| 29 | +end | ... | ... | 
plugins/mezuro/lib/kalibro/client/base_tool_client.rb
| ... | ... | @@ -1,16 +0,0 @@ | 
| 1 | -class Kalibro::Client::BaseToolClient | |
| 2 | - | |
| 3 | - def initialize | |
| 4 | - @port = Kalibro::Client::Port.new('BaseTool') | |
| 5 | - end | |
| 6 | - | |
| 7 | - def base_tool_names | |
| 8 | - @port.request(:get_base_tool_names)[:base_tool_name].to_a | |
| 9 | - end | |
| 10 | - | |
| 11 | - def base_tool(name) | |
| 12 | - hash = @port.request(:get_base_tool, {:base_tool_name => name})[:base_tool] | |
| 13 | - Kalibro::Entities::BaseTool.from_hash(hash) | |
| 14 | - end | |
| 15 | - | |
| 16 | -end | |
| 17 | 0 | \ No newline at end of file | 
plugins/mezuro/lib/kalibro/client/configuration_client.rb
| ... | ... | @@ -1,31 +0,0 @@ | 
| 1 | -class Kalibro::Client::ConfigurationClient | |
| 2 | - | |
| 3 | - def initialize | |
| 4 | - @port = Kalibro::Client::Port.new('Configuration') | |
| 5 | - end | |
| 6 | - | |
| 7 | - def save(configuration) | |
| 8 | - @port.request(:save_configuration, {:configuration => configuration.to_hash}) | |
| 9 | - end | |
| 10 | - | |
| 11 | - def self.save(configuration) | |
| 12 | - new.save(configuration) | |
| 13 | - end | |
| 14 | - | |
| 15 | - def configuration_names | |
| 16 | - @port.request(:get_configuration_names)[:configuration_name].to_a | |
| 17 | - end | |
| 18 | - | |
| 19 | - def configuration(name) | |
| 20 | - hash = @port.request(:get_configuration, {:configuration_name => name})[:configuration] | |
| 21 | - Kalibro::Entities::Configuration.from_hash(hash) | |
| 22 | - end | |
| 23 | - | |
| 24 | - def remove(configuration_name) | |
| 25 | - @port.request(:remove_configuration, {:configuration_name => configuration_name}) | |
| 26 | - end | |
| 27 | - | |
| 28 | - def self.remove(configuration_name) | |
| 29 | - new.remove(configuration_name) | |
| 30 | - end | |
| 31 | -end | 
plugins/mezuro/lib/kalibro/client/kalibro_client.rb
| ... | ... | @@ -1,19 +0,0 @@ | 
| 1 | -class Kalibro::Client::KalibroClient | |
| 2 | - | |
| 3 | - def initialize | |
| 4 | - @port = Kalibro::Client::Port.new('Kalibro') | |
| 5 | - end | |
| 6 | - | |
| 7 | - def supported_repository_types | |
| 8 | - @port.request(:get_supported_repository_types)[:repository_type].to_a | |
| 9 | - end | |
| 10 | - | |
| 11 | - def process_project(project_name) | |
| 12 | - @port.request(:process_project, {:project_name => project_name}) | |
| 13 | - end | |
| 14 | - | |
| 15 | - def self.process_project(project_name) | |
| 16 | - new.process_project(project_name) | |
| 17 | - end | |
| 18 | - | |
| 19 | -end | 
plugins/mezuro/lib/kalibro/client/module_result_client.rb
| ... | ... | @@ -1,27 +0,0 @@ | 
| 1 | -class Kalibro::Client::ModuleResultClient | |
| 2 | - | |
| 3 | - def initialize | |
| 4 | - @port = Kalibro::Client::Port.new('ModuleResult') | |
| 5 | - end | |
| 6 | - | |
| 7 | - def module_result(project_name, module_name, date) | |
| 8 | - hash = @port.request(:get_module_result, | |
| 9 | - {:project_name => project_name, :module_name => module_name, | |
| 10 | - :date => date_with_milliseconds(date)})[:module_result] | |
| 11 | - Kalibro::Entities::ModuleResult.from_hash(hash) | |
| 12 | - end | |
| 13 | - | |
| 14 | - def result_history(project_name, module_name) | |
| 15 | - value = @port.request(:get_result_history, | |
| 16 | - {:project_name => project_name, :module_name => module_name})[:module_result] | |
| 17 | - Kalibro::Entities::Entity.new.to_entity_array(value, Kalibro::Entities::ModuleResult) | |
| 18 | - end | |
| 19 | - | |
| 20 | - private | |
| 21 | - | |
| 22 | - def date_with_milliseconds(date) | |
| 23 | - milliseconds = "." + (date.sec_fraction * 60 * 60 * 24 * 1000).to_s | |
| 24 | - date.to_s[0..18] + milliseconds + date.to_s[19..-1] | |
| 25 | - end | |
| 26 | - | |
| 27 | -end | |
| 28 | 0 | \ No newline at end of file | 
plugins/mezuro/lib/kalibro/client/port.rb
| ... | ... | @@ -1,26 +0,0 @@ | 
| 1 | -require 'savon' | |
| 2 | - | |
| 3 | -Savon.configure do |config| | |
| 4 | - config.log = HTTPI.log = (RAILS_ENV == 'development') | |
| 5 | -end | |
| 6 | - | |
| 7 | -class Kalibro::Client::Port | |
| 8 | - | |
| 9 | - def initialize(endpoint) | |
| 10 | - @client = Savon::Client.new("#{service_address}#{endpoint}Endpoint/?wsdl") | |
| 11 | - end | |
| 12 | - | |
| 13 | - def service_address | |
| 14 | - if @service_address.nil? | |
| 15 | - service_file = "#{RAILS_ROOT}/plugins/mezuro/SERVICE" | |
| 16 | - File.open(service_file).each_line{ | line | @service_address = line } | |
| 17 | - end | |
| 18 | - @service_address | |
| 19 | - end | |
| 20 | - | |
| 21 | - def request(action, request_body = nil) | |
| 22 | - response = @client.request(:kalibro, action) { soap.body = request_body } | |
| 23 | - response.to_hash["#{action}_response".to_sym] | |
| 24 | - end | |
| 25 | - | |
| 26 | -end | 
plugins/mezuro/lib/kalibro/client/project_client.rb
| ... | ... | @@ -1,31 +0,0 @@ | 
| 1 | -class Kalibro::Client::ProjectClient | |
| 2 | - | |
| 3 | - def initialize | |
| 4 | - @port = Kalibro::Client::Port.new('Project') | |
| 5 | - end | |
| 6 | - | |
| 7 | - def save(project) | |
| 8 | - @port.request(:save_project, {:project => project.to_hash}) | |
| 9 | - end | |
| 10 | - | |
| 11 | - def self.save(project) | |
| 12 | - new.save(project) | |
| 13 | - end | |
| 14 | - | |
| 15 | - def project_names | |
| 16 | - @port.request(:get_project_names)[:project_name].to_a | |
| 17 | - end | |
| 18 | - | |
| 19 | - def project(name) | |
| 20 | - hash = @port.request(:get_project, {:project_name => name})[:project] | |
| 21 | - Kalibro::Entities::Project.from_hash(hash) | |
| 22 | - end | |
| 23 | - | |
| 24 | - def remove(project_name) | |
| 25 | - @port.request(:remove_project, {:project_name => project_name}) | |
| 26 | - end | |
| 27 | - | |
| 28 | - def self.remove(project_name) | |
| 29 | - new.remove(project_name) | |
| 30 | - end | |
| 31 | -end | 
plugins/mezuro/lib/kalibro/client/project_result_client.rb
| ... | ... | @@ -1,41 +0,0 @@ | 
| 1 | -class Kalibro::Client::ProjectResultClient | |
| 2 | - | |
| 3 | - def initialize | |
| 4 | - @port = Kalibro::Client::Port.new('ProjectResult') | |
| 5 | - end | |
| 6 | - | |
| 7 | - def has_results_for(project_name) | |
| 8 | - @port.request(:has_results_for, {:project_name => project_name})[:has_results] | |
| 9 | - end | |
| 10 | - | |
| 11 | - def has_results_before(project_name, date) | |
| 12 | - @port.request(:has_results_before, {:project_name => project_name, :date => date})[:has_results] | |
| 13 | - end | |
| 14 | - | |
| 15 | - def has_results_after(project_name, date) | |
| 16 | - @port.request(:has_results_after, {:project_name => project_name, :date => date})[:has_results] | |
| 17 | - end | |
| 18 | - | |
| 19 | - def first_result(project_name) | |
| 20 | - hash = @port.request(:get_first_result_of, {:project_name => project_name})[:project_result] | |
| 21 | - Kalibro::Entities::ProjectResult.from_hash(hash) | |
| 22 | - end | |
| 23 | - | |
| 24 | - def last_result(project_name) | |
| 25 | - hash = @port.request(:get_last_result_of, {:project_name => project_name})[:project_result] | |
| 26 | - Kalibro::Entities::ProjectResult.from_hash(hash) | |
| 27 | - end | |
| 28 | - | |
| 29 | - def first_result_after(project_name, date) | |
| 30 | - request_body = {:project_name => project_name, :date => date} | |
| 31 | - hash = @port.request(:get_first_result_after, request_body)[:project_result] | |
| 32 | - Kalibro::Entities::ProjectResult.from_hash(hash) | |
| 33 | - end | |
| 34 | - | |
| 35 | - def last_result_before(project_name, date) | |
| 36 | - request_body = {:project_name => project_name, :date => date} | |
| 37 | - hash = @port.request(:get_last_result_before, request_body)[:project_result] | |
| 38 | - Kalibro::Entities::ProjectResult.from_hash(hash) | |
| 39 | - end | |
| 40 | - | |
| 41 | -end | |
| 42 | 0 | \ No newline at end of file |