Mercurial > hg > soundsoftware-site
changeset 804:548e23d4cd71 live
Merge from branch "cannam"
author | Chris Cannam |
---|---|
date | Wed, 23 Nov 2011 11:15:54 +0000 |
parents | 37e39e437b2c (current diff) 6fa65a452888 (diff) |
children | 7429d10cc491 |
files | |
diffstat | 59 files changed, 2689 insertions(+), 92 deletions(-) [+] |
line wrap: on
line diff
--- a/app/controllers/application_controller.rb Tue Sep 20 10:57:26 2011 +0100 +++ b/app/controllers/application_controller.rb Wed Nov 23 11:15:54 2011 +0000 @@ -177,14 +177,14 @@ def find_project @project = Project.find(params[:id]) rescue ActiveRecord::RecordNotFound - render_404 + User.current.logged? ? render_404 : require_login end # Find project of id params[:project_id] def find_project_by_project_id @project = Project.find(params[:project_id]) rescue ActiveRecord::RecordNotFound - render_404 + User.current.logged? ? render_404 : require_login end # Find a project based on params[:project_id]
--- a/app/helpers/projects_helper.rb Tue Sep 20 10:57:26 2011 +0100 +++ b/app/helpers/projects_helper.rb Wed Nov 23 11:15:54 2011 +0000 @@ -150,9 +150,6 @@ if s != '' a = '' - a << "<h2>" - a << l("label_my_project_plural") - a << "</h2>" a << "<ul class='projects root'>\n" a << s a << "</ul>\n"
--- a/app/helpers/repositories_helper.rb Tue Sep 20 10:57:26 2011 +0100 +++ b/app/helpers/repositories_helper.rb Wed Nov 23 11:15:54 2011 +0000 @@ -281,4 +281,24 @@ ) + '<br />' + l(:text_scm_path_encoding_note)) end + + # Generates a link to a downloadable archive for a revision + # Options: + # * :text - Link text (default to the formatted revision) + def link_to_revision_archive(repository, revision, project, options={}) + method = repository.class.name.demodulize.underscore + "_link_to_revision_archive" + if repository.is_a?(Repository) && + respond_to?(method) && method != 'link_to_revision_archive' + send(method, repository, revision, project, options) + end + end + + def mercurial_link_to_revision_archive(repository, revision, project, options={}) + text = options.delete(:text) || format_revision(revision) + rev = revision.respond_to?(:identifier) ? revision.identifier : revision + if rev.blank? then rev = 'tip' end + content_tag('a', h(text), + { :href => "/hg/#{project.identifier}/archive/#{rev}.zip" }.merge(options)); + end + end
--- a/app/views/account/register.rhtml Tue Sep 20 10:57:26 2011 +0100 +++ b/app/views/account/register.rhtml Wed Nov 23 11:15:54 2011 +0000 @@ -29,9 +29,10 @@ <p><label for="user_mail"><%=l(:field_mail)%> <span class="required">*</span></label> <%= text_field 'user', 'mail' %></p> +<!-- We only support English in this site <p><label for="user_language"><%=l(:field_language)%></label> <%= select("user", "language", lang_options_for_select) %></p> - +--> <h3><%=l(:label_ssamr_details)%></h3>
--- a/app/views/repositories/_navigation.rhtml Tue Sep 20 10:57:26 2011 +0100 +++ b/app/views/repositories/_navigation.rhtml Wed Nov 23 11:15:54 2011 +0000 @@ -2,7 +2,7 @@ <%= javascript_include_tag 'repository_navigation' %> <% end %> -<%= link_to l(:label_statistics), {:action => 'stats', :id => @project}, :class => 'icon icon-stats' %> +<%= link_to_revision_archive(@repository, @changeset, @project, { :text => l(:label_download_revision), :class => 'icon icon-package' }) %> <% form_tag({:action => controller.action_name, :id => @project, :path => to_path_param(@path), :rev => ''}, {:method => :get, :id => 'revision_selector'}) do -%> <!-- Branches Dropdown --> @@ -19,3 +19,5 @@ | <%= l(:label_revision) %>: <%= text_field_tag 'rev', @rev, :size => 8 %> <% end -%> + +
--- a/app/views/repositories/revision.rhtml Tue Sep 20 10:57:26 2011 +0100 +++ b/app/views/repositories/revision.rhtml Wed Nov 23 11:15:54 2011 +0000 @@ -1,11 +1,15 @@ <div class="contextual"> + + <%= link_to_revision_archive(@repository, @changeset, @project, { :text => l(:label_download_revision), :class => 'icon icon-package' }) %> + + « <% unless @changeset.previous.nil? -%> <%= link_to_revision(@changeset.previous, @project, :text => l(:label_previous)) %> <% else -%> <%= l(:label_previous) %> <% end -%> -| + <% unless @changeset.next.nil? -%> <%= link_to_revision(@changeset.next, @project, :text => l(:label_next)) %> <% else -%> @@ -21,6 +25,7 @@ <%= text_field_tag 'rev', @rev, :size => 8 %> <%= submit_tag 'OK', :name => nil %> <% end %> + </div> <h2><%= l(:label_revision) %> <%= format_revision(@changeset) %></h2>
--- a/app/views/repositories/show.rhtml Tue Sep 20 10:57:26 2011 +0100 +++ b/app/views/repositories/show.rhtml Wed Nov 23 11:15:54 2011 +0000 @@ -51,9 +51,14 @@ :id => @project, :page => nil, :key => User.current.rss_key})) %> <% end %> +<p class="statistics"> +<%= link_to l(:label_statistics), {:action => 'stats', :id => @project}, :class => 'icon icon-stats' %> +</p> + <% other_formats_links do |f| %> <%= f.link_to 'Atom', :url => {:action => 'revisions', :id => @project, :key => User.current.rss_key} %> <% end %> + <% end %> <% end %>
--- a/config/locales/en.yml Tue Sep 20 10:57:26 2011 +0100 +++ b/config/locales/en.yml Wed Nov 23 11:15:54 2011 +0000 @@ -169,7 +169,7 @@ notice_failed_to_save_issues: "Failed to save %{count} issue(s) on %{total} selected: %{ids}." notice_failed_to_save_members: "Failed to save member(s): %{errors}." notice_no_issue_selected: "No issue is selected! Please, check the issues you want to edit." - notice_account_pending: "Your account was created and is now pending administrator approval." + notice_account_pending: "Your account is now awaiting administrator approval. You will receive a notification email when your account has been activated." notice_default_data_loaded: Default configuration successfully loaded. notice_unable_delete_version: Unable to delete version. notice_unable_delete_time_entry: Unable to delete time log entry. @@ -698,6 +698,7 @@ label_latest_revision_plural: Latest revisions label_view_revisions: View revisions label_view_all_revisions: View all revisions + label_download_revision: Download as Zip label_max_size: Maximum size label_sort_highest: Move to top label_sort_higher: Move up
--- a/extra/soundsoftware/SoundSoftware.pm Tue Sep 20 10:57:26 2011 +0100 +++ b/extra/soundsoftware/SoundSoftware.pm Wed Nov 23 11:15:54 2011 +0000 @@ -110,6 +110,11 @@ req_override => OR_AUTHCFG, args_how => TAKE1, }, + { + name => 'SoundSoftwareSslRequired', + req_override => OR_AUTHCFG, + args_how => TAKE1, + }, ); sub SoundSoftwareDSN { @@ -143,6 +148,8 @@ } } +sub SoundSoftwareSslRequired { set_val('SoundSoftwareSslRequired', @_); } + sub trim { my $string = shift; $string =~ s/\s{2,}/ /g; @@ -184,33 +191,80 @@ my $project_id = get_project_identifier($dbh, $r); - if (!defined $read_only_methods{$method}) { - print STDERR "SoundSoftware.pm:$$: Method is not read-only\n"; - if (project_repo_is_readonly($dbh, $project_id, $r)) { - print STDERR "SoundSoftware.pm:$$: Project repo is read-only, refusing access\n"; - return FORBIDDEN; - } else { - print STDERR "SoundSoftware.pm:$$: Project repo is read-write, authentication handler required\n"; - return OK; - } - } + # We want to delegate most of the work to the authentication + # handler (to ensure that user is asked to login even for + # nonexistent projects -- so they can't tell whether a private + # project exists or not without authenticating). So + # + # * if the project is public + # - if the method is read-only + # + set handler to OK, no auth needed + # - if the method is not read-only + # + if the repo is read-only, return forbidden + # + else require auth + # * if the project is not public or does not exist + # + require auth + # + # If we are requiring auth and are not currently https, and + # https is required, then we must return a redirect to https + # instead of an OK. my $status = get_project_status($dbh, $project_id, $r); + my $readonly = project_repo_is_readonly($dbh, $project_id, $r); $dbh->disconnect(); undef $dbh; - if ($status == 0) { # nonexistent - print STDERR "SoundSoftware.pm:$$: Project does not exist, refusing access\n"; - return FORBIDDEN; - } elsif ($status == 1) { # public - print STDERR "SoundSoftware.pm:$$: Project is public, no restriction here\n"; - $r->set_handlers(PerlAuthenHandler => [\&OK]) - } else { # private - print STDERR "SoundSoftware.pm:$$: Project is private, authentication handler required\n"; + my $auth_ssl_reqd = will_require_ssl_auth($r); + + if ($status == 1) { # public + + print STDERR "SoundSoftware.pm:$$: Project is public\n"; + + if (!defined $read_only_methods{$method}) { + + print STDERR "SoundSoftware.pm:$$: Method is not read-only\n"; + + if ($readonly) { + print STDERR "SoundSoftware.pm:$$: Project repo is read-only, refusing access\n"; + return FORBIDDEN; + } else { + print STDERR "SoundSoftware.pm:$$: Project repo is read-write, auth required\n"; + # fall through, this is the normal case + } + + } elsif ($auth_ssl_reqd and $r->unparsed_uri =~ m/cmd=branchmap/) { + + # A hac^H^H^Hspecial case. We want to ensure we switch to + # https (if it will be necessarily for authentication) + # before the first POST request, and this is what I think + # will give us suitable warning for Mercurial. + + print STDERR "SoundSoftware.pm:$$: Switching to HTTPS in preparation\n"; + # fall through, this is the normal case + + } else { + # Public project, read-only method -- this is the only + # case we can decide for certain to accept in this function + print STDERR "SoundSoftware.pm:$$: Method is read-only, no restriction here\n"; + $r->set_handlers(PerlAuthenHandler => [\&OK]); + return OK; + } + + } else { # status != 1, i.e. nonexistent or private -- equivalent here + + print STDERR "SoundSoftware.pm:$$: Project is private or nonexistent, auth required\n"; + # fall through } - return OK + if ($auth_ssl_reqd) { + my $redir_to = "https://" . $r->hostname() . $r->unparsed_uri(); + print STDERR "SoundSoftware.pm:$$: Need to switch to HTTPS, redirecting to $redir_to\n"; + $r->headers_out->add('Location' => $redir_to); + return REDIRECT; + } else { + return OK; + } } sub authen_handler { @@ -237,6 +291,16 @@ print STDERR "SoundSoftware.pm:$$: User is " . $r->user . ", got password\n"; + my $status = get_project_status($dbh, $project_id, $r); + if ($status == 0) { + # nonexistent, behave like private project you aren't a member of + print STDERR "SoundSoftware.pm:$$: Project doesn't exist, not permitted\n"; + $dbh->disconnect(); + undef $dbh; + $r->note_auth_failure(); + return AUTH_REQUIRED; + } + my $permitted = is_permitted($dbh, $project_id, $r->user, $redmine_pass, $r); $dbh->disconnect(); @@ -279,6 +343,30 @@ $ret; } +sub will_require_ssl_auth { + my $r = shift; + + my $cfg = Apache2::Module::get_config + (__PACKAGE__, $r->server, $r->per_dir_config); + + if ($cfg->{SoundSoftwareSslRequired} eq "on") { + if ($r->dir_config('HTTPS') eq "on") { + # already have ssl + return 0; + } else { + # require ssl for auth, don't have it yet + return 1; + } + } elsif ($cfg->{SoundSoftwareSslRequired} eq "off") { + # don't require ssl for auth + return 0; + } else { + print STDERR "WARNING: SoundSoftware.pm:$$: SoundSoftwareSslRequired should be either 'on' or 'off'\n"; + # this is safer + return 1; + } +} + sub project_repo_is_readonly { my $dbh = shift; my $project_id = shift; @@ -368,6 +456,7 @@ } $sthldap->finish(); undef $sthldap; + last if ($ret); } } else { print STDERR "SoundSoftware.pm:$$: User $redmine_user lacks required role for this project\n"; @@ -383,14 +472,13 @@ sub get_project_identifier { my $dbh = shift; my $r = shift; - my $location = $r->location; - my ($repo) = $r->uri =~ m{$location/*([^/]+)}; + my ($repo) = $r->uri =~ m{$location/*([^/]*)}; return $repo if (!$repo); $repo =~ s/[^a-zA-Z0-9\._-]//g; - + # The original Redmine.pm returns the string just calculated as # the project identifier. That won't do for us -- we may have # (and in fact already do have, in our test instance) projects @@ -410,7 +498,6 @@ my $prefix = $cfg->{SoundSoftwareRepoPrefix}; if (!defined $prefix) { $prefix = '%/'; } - my $identifier = ''; $sth->execute($prefix . $repo); @@ -449,6 +536,18 @@ # to project identifier if any are found if ($name =~ m/[^\w\d\s\._-]/) { $name = $project_id; + } elsif ($name =~ m/^\s*$/) { + # empty or whitespace + $name = $project_id; + } + + if ($name =~ m/^\s*$/) { + # nothing even in $project_id -- probably a nonexistent project. + # use repo name instead (don't want to admit to user that project + # doesn't exist) + my $location = $r->location; + my ($repo) = $r->uri =~ m{$location/*([^/]*)}; + $name = $repo; } my $realm = '"Mercurial repository for ' . "'$name'" . '"';
--- a/public/help/wiki_syntax_detailed.html Tue Sep 20 10:57:26 2011 +0100 +++ b/public/help/wiki_syntax_detailed.html Wed Nov 23 11:15:54 2011 +0000 @@ -68,7 +68,7 @@ <p>Wiki links are displayed in red if the page doesn't exist yet, eg: <a href="#" class="wiki-page new">Nonexistent page</a>.</p> - <p>Links to other resources:</p> + <p>Links to other resources:</p> <ul> <li>Documents: @@ -76,7 +76,7 @@ <li><strong>document#17</strong> (link to document with id 17)</li> <li><strong>document:Greetings</strong> (link to the document with title "Greetings")</li> <li><strong>document:"Some document"</strong> (double quotes can be used when document title contains spaces)</li> - <li><strong>sandbox:document:"Some document"</strong> (link to a document with title "Some document" in other project "sandbox")</li> + <li><strong>sandbox:document:"Some document"</strong> (link to a document with title "Some document" in other project "sandbox")</li> </ul></li> </ul> @@ -99,38 +99,38 @@ </ul> <ul> - <li>Repository files: + <li>Repository files: <ul> - <li><strong>source:some/file</strong> (link to the file located at /some/file in the project's repository)</li> - <li><strong>source:some/file@52</strong> (link to the file's revision 52)</li> - <li><strong>source:some/file#L120</strong> (link to line 120 of the file)</li> - <li><strong>source:some/file@52#L120</strong> (link to line 120 of the file's revision 52)</li> - <li><strong>source:"some file@52#L120"</strong> (use double quotes when the URL contains spaces</li> + <li><strong>source:some/file</strong> (link to the file located at /some/file in the project's repository)</li> + <li><strong>source:some/file@52</strong> (link to the file's revision 52)</li> + <li><strong>source:some/file#L120</strong> (link to line 120 of the file)</li> + <li><strong>source:some/file@52#L120</strong> (link to line 120 of the file's revision 52)</li> + <li><strong>source:"some file@52#L120"</strong> (use double quotes when the URL contains spaces</li> <li><strong>export:some/file</strong> (force the download of the file)</li> <li><strong>sandbox:source:some/file</strong> (link to the file located at /some/file in the repository of the project "sandbox")</li> - <li><strong>sandbox:export:some/file</strong> (force the download of the file)</li> - </ul></li> + <li><strong>sandbox:export:some/file</strong> (force the download of the file)</li> + </ul></li> </ul> <ul> - <li>Forum messages: - <ul> - <li><strong>message#1218</strong> (link to message with id 1218)</li> - </ul></li> - </ul> - - <ul> - <li>Projects: - <ul> - <li><strong>project#3</strong> (link to project with id 3)</li> - <li><strong>project:someproject</strong> (link to project named "someproject")</li> - </ul></li> - </ul> - - - <p>Escaping:</p> - - <ul> + <li>Forum messages: + <ul> + <li><strong>message#1218</strong> (link to message with id 1218)</li> + </ul></li> + </ul> + + <ul> + <li>Projects: + <ul> + <li><strong>project#3</strong> (link to project with id 3)</li> + <li><strong>project:someproject</strong> (link to project named "someproject")</li> + </ul></li> + </ul> + + + <p>Escaping:</p> + + <ul> <li>You can prevent Redmine links from being parsed by preceding them with an exclamation mark: !</li> </ul> @@ -198,6 +198,78 @@ <p>Redmine assigns an anchor to each of those headings thus you can link to them with "#Heading", "#Subheading" and so forth.</p> + + +<h3><a name="9" class="wiki-page"></a>Bullets and Numbering</h3> + +<pre> +* First Level Bullet +** Second Level Bullet +** Another Second Level Bullet +*** Third Level Bullet +** Back to 2nd Level Bullet +* Back to 1st Level Bullet +</pre> + +<ul> + <li>First Level Bullet</li> + <ul> + <li>Second Level Bullet</li> + <li>Another Second Level Bullet</li> + <ul> + <li>Third Level Bullet</li> + </ul> + <li>Back to 2nd Level Bullet</li> + </ul> + + <li>Back to 1st Level Bullet</li> +</ul> + +<pre> +# First Level Numbering +## Second Level Numbering +## Another Second Level Numbering +### Third Level Numbering +## Back to 2nd Level Numbering +# Back to 1st Level Numbering +</pre> + +<ol> + <li>First Level Numbering</li> + <ol> + <li>Second Level Numbering</li> + <li>Another Second Level Numbering</li> + <ol> + <li>Third Level Numbering</li> + </ol> + <li>Back to 2nd Level Numbering</li> + </ol> + <li>Back to 1st Level Numbering</li> +</ol> + +<pre> +# First Level Numbering +#* Bullet inside numbering environment +#* Another Bullet inside numbering environment +# Back to 1st Level Numbering +</pre> + +<ol> + <li>First Level Numbering</li> + <ul> + <li>Bullet inside numbering environment</li> + <li>Another Bullet inside numbering environment</li> + </ul> + + <li>Back to 1st Level Numbering</li> +</ol> + + + + + + + <h3><a name="9" class="wiki-page"></a>Paragraphs</h3> <pre> @@ -242,7 +314,7 @@ <h2><a name="13" class="wiki-page"></a>Code highlighting</h2> - <p>Code highlightment relies on <a href="http://coderay.rubychan.de/" class="external">CodeRay</a>, a fast syntax highlighting library written completely in Ruby. It currently supports c, cpp, css, delphi, groovy, html, java, javascript, json, php, python, rhtml, ruby, scheme, sql, xml and yaml languages.</p> + <p>Code highlightment relies on <a href="http://coderay.rubychan.de/" class="external">CodeRay</a>, a fast syntax highlighting library written completely in Ruby. It currently supports c, cpp, css, delphi, groovy, html, java, javascript, json, php, python, rhtml, ruby, scheme, sql, xml and yaml languages.</p> <p>You can highlight code in your wiki page using this syntax:</p>
--- a/public/themes/soundsoftware/stylesheets/application.css Tue Sep 20 10:57:26 2011 +0100 +++ b/public/themes/soundsoftware/stylesheets/application.css Wed Nov 23 11:15:54 2011 +0000 @@ -134,6 +134,8 @@ #footer { background-color: #fdfbf5; border: 0; border-top: 2px solid #a9b680; color: #3e442c; text-align: right; } #footer a { color: #be5700; font-weight: bold; } +p.statistics { text-align: right; font-size:0.9em; color: #666; } + #main { margin-top: 135px; font:95%; background: #fdfaf0; } #main a { font-weight: medium; color: #be5700;} #main a:hover { color: #be5700; text-decoration: underline; } @@ -224,5 +226,16 @@ .embedded #matlabdoc th { text-align: left; } +/* autocomplete positioning fix */ +div.autocomplete { + margin-top:136px; +} +#my_projects_fieldset.collapsible { + font-size: 1.0em; + font-color: green; +} + + +
--- a/vendor/plugins/redmine_bibliography/app/helpers/publications_helper.rb Tue Sep 20 10:57:26 2011 +0100 +++ b/vendor/plugins/redmine_bibliography/app/helpers/publications_helper.rb Wed Nov 23 11:15:54 2011 +0000 @@ -12,8 +12,12 @@ def projects_check_box_tags(name, projects) s = '' projects.sort.each do |project| - s << "<label>#{ check_box_tag name, project.id, false } #{link_to_project project}</label>\n" + if User.current.allowed_to?(:edit_publication, project) + s << "<label>#{ check_box_tag name, project.id, false } #{link_to_project project}</label>\n" + s << '<br />' + end end + s end @@ -43,12 +47,12 @@ f.hidden_field(:_destroy) + link_to_function(name, "remove_fields(this)", :class => 'icon icon-del') end - def link_to_add_fields(name, f, association) + def link_to_add_author_fields(name, f, association, action) new_object = f.object.class.reflect_on_association(association).klass.new fields = f.fields_for(association, new_object, :child_index => "new_#{association}") do |builder| render(association.to_s.singularize + "_fields", :f => builder) end - link_to_function(name, h("add_fields(this, '#{association}', '#{escape_javascript(fields)}')"), { :class => 'icon icon-add', :id => "add_another_author" }) + link_to_function(name, h("add_author_fields(this, '#{association}', '#{escape_javascript(fields)}', '#{action}')"), { :class => 'icon icon-add', :id => "add_another_author" }) end def sanitized_object_name(object_name)
--- a/vendor/plugins/redmine_bibliography/app/views/projects/_bibliography_box.html.erb Tue Sep 20 10:57:26 2011 +0100 +++ b/vendor/plugins/redmine_bibliography/app/views/projects/_bibliography_box.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -7,18 +7,16 @@ <dl> <% @project.publications.each do |publication| %> <dt> - <%= link_to publication.title, :controller => 'publications', :action => 'show', :id => publication, :project_id => @project %> - </dt> - <dd> <span class="authors"> - <%= publication.authorships.map { |a| h a.name_on_paper }.join(', ') %> + <%= publication.authorships.map { |a| h a.name_on_paper }.join(', ') %><% if !publication.authorships.empty? %>.<% end %> </span> + <span class="title"><%= link_to publication.title, :controller => 'publications', :action => 'show', :id => publication, :project_id => @project %></span> <% if publication.bibtex_entry.year.to_s != "" %> <span class="year"> - <%= publication.bibtex_entry.year %> + (<%= publication.bibtex_entry.year %>) </span> <% end %> - </dd> + </dt><dd></dd> <% end -%> </dl> </div>
--- a/vendor/plugins/redmine_bibliography/app/views/publications/_authorship_fields.rhtml Tue Sep 20 10:57:26 2011 +0100 +++ b/vendor/plugins/redmine_bibliography/app/views/publications/_authorship_fields.rhtml Wed Nov 23 11:15:54 2011 +0000 @@ -19,11 +19,11 @@ <p style="margin-bottom: -2.5em; padding-bottom; 0"><label><%= l(:identify_author_question) %></label></p> <p class="author_identify"> - <label class='inline'><%= radio_button_tag(:identify_author, "yes", false, :name => form_tag_name(f.object_name,:identify_author ), :id => form_tag_id( f.object_name, :identify_author_yes ), :onclick => "identify_author_status($(this).value, #{form_object_id(f.object_name) });") %> <%= l(:identify_author_yes) %> </label><br /> + <label class='inline'><%= radio_button_tag(:identify_author, "yes", false, :name => form_tag_name(f.object_name,:identify_author ), :id => form_tag_id( f.object_name, :identify_author_yes ), :onchange => "identify_author_status($(this).value, #{form_object_id(f.object_name) });") %> <%= l(:identify_author_yes) %> </label><br /> - <label class='inline'><%= radio_button_tag(:identify_author, "correct", false, :name => form_tag_name(f.object_name,:identify_author ), :id => form_tag_id( f.object_name, :identify_author_corrections ), :onclick => "identify_author_status($(this).value, #{form_object_id(f.object_name) });") %> <%= l(:identify_author_correct) %> </label><br /> + <label class='inline'><%= radio_button_tag(:identify_author, "correct", false, :name => form_tag_name(f.object_name,:identify_author ), :id => form_tag_id( f.object_name, :identify_author_corrections ), :onchange => "identify_author_status($(this).value, #{form_object_id(f.object_name) });") %> <%= l(:identify_author_correct) %> </label><br /> - <label class='inline'><%= radio_button_tag(:identify_author, "no", true, :name => form_tag_name(f.object_name,:identify_author ), :id => form_tag_id( f.object_name, :identify_author_no ), :onclick => "identify_author_status($(this).value, #{form_object_id(f.object_name) });") %> <%= l(:identify_author_no) %> </label><br /> + <label class='inline'><%= radio_button_tag(:identify_author, "no", true, :name => form_tag_name(f.object_name,:identify_author ), :id => form_tag_id( f.object_name, :identify_author_no ), :onchange => "identify_author_status($(this).value, #{form_object_id(f.object_name) });") %> <%= l(:identify_author_no) %> </label><br /> </p> </div> @@ -45,12 +45,11 @@ <p> - <% if params[:action] == 'new' %> - <%= button_to_function l(:label_save_author), {}, { :onclick => "toggle_save_author(#{form_object_id(f.object_name)}); return false;", :id => form_tag_id( f.object_name, :edit_save_button )} %> - <% else %> -<%= button_to_function l(:label_edit_author), {}, { :onclick => "toggle_save_author(#{form_object_id(f.object_name)}); return false;", :id => form_tag_id( f.object_name, :edit_save_button )} %> - - <% end %> + <%- if params[:action] == 'new' -%> + <%= button_to_function l(:label_save_author), {}, { :onclick => "toggle_save_author(#{form_object_id(f.object_name)}); return false;", :id => form_tag_id( f.object_name, :edit_save_button )} %> + <%- else -%> + <%= button_to_function l(:label_edit_author), {}, { :onclick => "toggle_save_author(#{form_object_id(f.object_name)}); return false;", :id => form_tag_id( f.object_name, :edit_save_button )} %> + <%- end -%> <%= link_to_remove_fields l("remove_author"), f %>
--- a/vendor/plugins/redmine_bibliography/app/views/publications/_form.html.erb Tue Sep 20 10:57:26 2011 +0100 +++ b/vendor/plugins/redmine_bibliography/app/views/publications/_form.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -24,7 +24,7 @@ <% f.fields_for :authorships do |builder| -%> <%= render "authorship_fields", :f => builder %> <%- end -%> - <%= link_to_add_fields l(:label_add_an_author), f, :authorships %> + <%= link_to_add_author_fields l(:label_add_an_author), f, :authorships, params[:action] %> </div> </div>
--- a/vendor/plugins/redmine_bibliography/app/views/users/show.rhtml Tue Sep 20 10:57:26 2011 +0100 +++ b/vendor/plugins/redmine_bibliography/app/views/users/show.rhtml Wed Nov 23 11:15:54 2011 +0000 @@ -51,18 +51,17 @@ <% @publications.each do |publication|%> <dt> - <%= link_to publication.title, :controller => 'publications', :action => 'show', :id => publication %> - </dt> - - <dd> <span class="authors"> - <%= publication.authorships.map { |a| h a.name_on_paper }.join(', ') %> + <%= publication.authorships.map { |a| h a.name_on_paper }.join(', ') %><% if !publication.authorships.empty? %>.<% end %> </span> + <span class="title"><%= link_to publication.title, :controller => 'publications', :action => 'show', :id => publication %></span> <% if publication.bibtex_entry.year.to_s != "" %> <span class="year"> - <%= publication.bibtex_entry.year %> + (<%= publication.bibtex_entry.year %>) </span> <% end %> + </dt> + <dd> </dd> <% end %> </div>
--- a/vendor/plugins/redmine_bibliography/assets/javascripts/authors.js Tue Sep 20 10:57:26 2011 +0100 +++ b/vendor/plugins/redmine_bibliography/assets/javascripts/authors.js Wed Nov 23 11:15:54 2011 +0000 @@ -3,12 +3,15 @@ $(link).up(".fields").hide(); } -function add_fields(link, association, content) { - var new_id = new Date().getTime(); - var regexp = new RegExp("new_" + association, "g") - $(link).insert({ - before: content.replace(regexp, new_id) - }); +function add_author_fields(link, association, content, action) { + var new_id = new Date().getTime(); + var regexp = new RegExp("new_" + association, "g"); + $(link).insert({ + before: content.replace(regexp, new_id) + }); + if(action != "new"){ + toggle_save_author(new_id, $(link)); + }; } function identify_author_status(status, object_id) {
--- a/vendor/plugins/redmine_bibliography/assets/stylesheets/bibliography.css Tue Sep 20 10:57:26 2011 +0100 +++ b/vendor/plugins/redmine_bibliography/assets/stylesheets/bibliography.css Wed Nov 23 11:15:54 2011 +0000 @@ -39,6 +39,7 @@ div#bibliography dd { margin-bottom: 1em; padding-left: 18px; font-size: 0.9em; } div#bibliography .box dd { margin-bottom: 0.6em; padding-left: 0; } div#bibliography dd .authors { font-style: italic; } +div#bibliography dt .title { font-style: italic; } div#bibliography dd span.authors { color: #808080; } div#bibliography dd span.year { padding-left: 0.6em; }
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/COPYING Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,17 @@ +redmine_tags is a redMine plugin, that adds tagging support. + +Copyright (c) 2010 Eric Davis +Copyright (c) 2010 Aleksey V Zapparov AKA ixti + +redmine_tags is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +redmine_tags is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +You should have received a copy of the GNU General Public License +along with redmine_tags. If not, see <http://www.gnu.org/licenses/>.
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/LICENSE Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,619 @@ + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/> + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee.
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/README.rdoc Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,15 @@ += Redmine Tags + +Allows marking up different models in Redmine with tags. +Inspired by original redmine_tags of Eric Davis. But in +comparison extended with some code that was already wrote +as part of my own redmine taggable_issues branch. + +== Supported models + +* Issues + +== License + +This plugin is licensed under the GNU/GPL v3. +See COPYING and LICENSE for details.
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/helpers/filters_helper.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,62 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +module FiltersHelper + # returns link to the page with issues filtered by specified filters + # === parameters + # * <i>title</i> = link title text + # * <i>filters</i> = filters to be applied (see <tt>link_to_filter_options</tt> for details) + # * <i>options</i> = (optional) base options of the link + # === example + # link_to_filter 'foobar', [[ :tags, '~', 'foobar' ]] + # link_to_filter 'foobar', [[ :tags, '~', 'foobar' ]], :project_id => project + def link_to_filter(title, filters, options = {}) + options.merge! link_to_filter_options(filters) + link_to title, options + end + + + # returns hash suitable for passing it to the <tt>to_link</tt> + # === parameters + # * <i>filters</i> = array of arrays. each child array is an array of strings: + # name, operator and value + # === example + # link_to 'foobar', link_to_filter_options [[ :tags, '~', 'foobar' ]] + # + # filters = [[ :tags, '~', 'bazbaz' ], [:status_id, 'o']] + # link_to 'bazbaz', link_to_filter_options filters + def link_to_filter_options(filters) + options = { + :controller => 'issues', + :action => 'index', + :set_filter => 1, + :fields => [], + :values => {}, + :operators => {} + } + + filters.each do |f| + name, operator, value = f + options[:fields].push(name) + options[:operators][name] = operator + options[:values][name] = [value] + end + + options + end +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/helpers/tags_helper.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,94 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +module TagsHelper + include ActsAsTaggableOn::TagsHelper + include FiltersHelper + + + # Returns tag link + # === Parameters + # * <i>tag</i> = Instance of Tag + # * <i>options</i> = (optional) Options (override system settings) + # * show_count - Boolean. Whenever show tag counts + # * open_only - Boolean. Whenever link to the filter with "open" issues + # only limit. + def render_tag_link(tag, options = {}) + filters = [[:tags, '=', tag.name]] + filters << [:status_id, 'o'] if options[:open_only] + + content = link_to_filter tag.name, filters, :project_id => @project + if options[:show_count] + content << content_tag('span', "(#{tag.count})", :class => 'tag-count') + end + + content_tag('span', content, :class => 'tag-label') + end + + def render_project_tag_link(tag) + content = link_to tag.name, :controller => :projects, :action => :index, :project => { :tag_list => tag.name } + content_tag('span', content, :class => 'tag-label') + end + + + # Renders list of tags + # Clouds are rendered as block <tt>div</tt> with internal <tt>span</t> per tag. + # Lists are rendered as unordered lists <tt>ul</tt>. Lists are ordered by + # <tt>tag.count</tt> descending. + # === Parameters + # * <i>tags</i> = Array of Tag instances + # * <i>options</i> = (optional) Options (override system settings) + # * show_count - Boolean. Whenever show tag counts + # * open_only - Boolean. Whenever link to the filter with "open" issues + # only limit. + # * style - list, cloud + def render_tags_list(tags, options = {}) + unless tags.nil? or tags.empty? + content, style = '', options.delete(:style) + + tags.sort! { |a,b| b.count <=> a.count } + + if :list == style + list_el, item_el = 'ul', 'li' + elsif :cloud == style + list_el, item_el = 'div', 'span' + tags = cloudify(tags) + else + raise "Unknown list style" + end + + tag_cloud tags, (1..8).to_a do |tag, weight| + content << " " + content_tag(item_el, render_tag_link(tag, options), :class => "tag-nube-#{weight}") + " " + end + + content_tag(list_el, content, :class => 'tags') + end + end + + private + # put most massive tags in the middle + def cloudify(tags) + temp, tags, trigger = tags, [], true + temp.each do |tag| + tags.send((trigger ? 'push' : 'unshift'), tag) + trigger = !trigger + end + tags + end + +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/auto_completes/_search_tag_list.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,5 @@ +<ul> + <% @tags.each do |tag| -%> + <%= content_tag 'li', h('%s (%d)' % [tag.name, tag.count]), :name => tag.name %> + <% end -%> +</ul>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/auto_completes/_tag_list.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,6 @@ +<ul> +<% @tags.each do |tag| -%> + <%= content_tag 'li', h('%s (%d)' % [tag.name, tag.count]), :name => tag.name %> +<% end -%> + <%= content_tag 'li', l(:auto_complete_new_tag) % @name, :name => @name %> +</ul>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/issues/_tags.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,6 @@ +<% unless issue.tag_list.empty? %> + <tr> + <td><b><%=l(:tags)%>:</b></td> + <td><%= issue.tag_counts.collect{ |t| render_tag_link(t, :show_count => false, :open_only => false) }.join(', ') %></td> + </tr> +<% end %>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/issues/_tags_form.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,8 @@ +<% fields_for :issue, issue, :builder => TabularFormBuilder do |f| -%> +<div> + <p id="issue_tags"><%= f.text_field :tag_list, :label => :tags, :size => 60, :class => 'hol' %></p> + <div id="issue_tag_candidates" class="autocomplete"></div> + <%= javascript_include_tag 'tags_input', :plugin => 'redmine_tags' %> + <%= javascript_tag "observeIssueTagsField('#{url_for(:controller => 'auto_completes', :action => 'issue_tags', :project_id => issue.project)}')" %> +</div> +<% end -%>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/issues/_tags_sidebar.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,5 @@ +<% unless sidebar_tags.empty? -%> + <%= stylesheet_link_tag 'redmine_tags', :plugin => 'redmine_tags' %> + <h3><%= l(:tags) %></h3> + <%= render_sidebar_tags %> +<% end -%>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/projects/_filter_search_tags.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,13 @@ +<p class='tag'> + <% fields_for @project, :builder => TabularFormBuilder do |f| -%> + <div> + <p id="project_tags"> + <%= f.text_field :tag_list, :label => :label_tags_search, :size => 60, :class => 'hol' %> + </p> + <div id="project_tag_candidates" class="autocomplete"></div> + <%= javascript_include_tag 'tags_input', :plugin => 'redmine_tags' %> + + <%= javascript_tag "observeProjectTagsField('#{url_for(:controller => 'auto_completes', :action => 'project_search_tags', :project_id => Project.first.id)}', true)" %> + </div> + <% end -%> +</p>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/projects/_filter_tags.rhtml Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,13 @@ +<p class='tag'> + <% fields_for @project, :builder => TabularFormBuilder do |f| -%> + <div> + <p id="project_tags"> + <%= f.text_field :tag_list, :label => :tags, :size => 60, :class => 'hol' %> + </p> + <div id="project_tag_candidates" class="autocomplete"></div> + <%= javascript_include_tag 'tags_input', :plugin => 'redmine_tags' %> + + <%= javascript_tag "observeProjectTagsField('#{url_for(:controller => 'auto_completes', :action => 'project_tags', :project_id => Project.first.id)}')" %> + </div> + <% end -%> +</p>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/projects/_filtered_projects.rhtml Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,7 @@ +<% if @projects.empty? %> +<p><b><%= l(:project_filter_no_results) %></b></p> +<% else %> +<%= render_project_table_with_filtering(@projects, @question) %> +<% end %> + +<p class="pagination"><%= pagination_links_full @project_pages, @project_count %></p>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/projects/_my_projects.rhtml Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,14 @@ +<% if @myproj_status=="true" %> +<fieldset id="my_projects_fieldset" class="collapsible"> + <legend onclick="toggleFieldsetWithState(this);"><h2><%= l(:label_my_project_plural) %></h2></legend> +<% else %> +<fieldset id="my_projects_fieldset" class="collapsible collapsed"> + <legend onclick="toggleFieldsetWithState(this);"><h2><%= l(:label_my_project_plural) %></h2></legend> + <div style="display: none;"> +<% end%> + <% if @user_projects %> + <div> + <%= render_my_project_hierarchy_with_tags(@user_projects)%> + </div> + <% end %> +</fieldset>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/projects/_tags.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,6 @@ +<% unless @project.tag_list.empty? %> + <tr> + <td><b><%=l(:tags)%>:</b></td> + <td><%= @project.tag_counts.collect{ |t| render_project_tag_link(t) }.join(', ') %></td> + </tr> +<% end %>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/projects/_tags_form.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,8 @@ +<% fields_for :project, project, :builder => TabularFormBuilder do |f| -%> +<div> + <p id="project_tags"><%= f.text_field :tag_list, :label => :tags, :size => 60, :class => 'hol' %></p> + <div id="project_tag_candidates" class="autocomplete"></div> + <%= javascript_include_tag 'tags_input', :plugin => 'redmine_tags' %> + <%= javascript_tag "observeProjectTagsField('#{url_for(:controller => 'auto_completes', :action => 'project_tags', :project_id => project)}', false)" %> +</div> +<% end -%> \ No newline at end of file
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/projects/index.rhtml Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,64 @@ +<% content_for :header_tags do %> + <%= auto_discovery_link_tag(:atom, {:action => 'index', :format => 'atom', :key => User.current.rss_key}) %> +<% end %> +<%= javascript_include_tag 'projects_index', :plugin => 'redmine_tags' %> + + +<div class="contextual"> + <%= link_to l(:label_overall_activity), { :controller => 'activities', :action => 'index' }%> + <%= '| ' + link_to(l(:label_project_new), {:controller => 'projects', :action => 'new'}, :class => 'icon icon-add') if User.current.allowed_to?(:add_project, nil, :global => true) %> +</div> + + + +<div style="clear:both;"></div> +<% if User.current.logged? %> + <%= render :partial => 'my_projects' %> +<% end %> + +<div style="clear:both;"></div> +<h2> + <%= l("label_project_all") %> +</h2> + +<div style="clear:both;"></div> +<% form_remote_tag(:controller => :projects, :action => :index, :method => :get, :html => {:id => :project_filtering_form}) do %> + +<% if @filter_status=="true" %> +<fieldset id="filters_fieldset" class="collapsible"> + <legend onclick="toggleFieldsetWithState(this);"><%= l(:label_filter_plural) %></legend> +<% else %> + <fieldset id="filters_fieldset" class="collapsible collapsed"> + <legend onclick="toggleFieldsetWithState(this);"><%= l(:label_filter_plural) %></legend> + <div style="display: none;"> +<% end %> + <div> + + <div id='filter_tags'> + <%= render :partial => 'filter_search_tags' %> + </div> + + <p class='q'> + <%= label_tag 'q', l('project_filtering_q_label') %> + <%= text_field_tag 'q', @question, :size => 30, :id => 'search-input' %> + </p> + + <p style="display: none;"><%= submit_tag( l('button_filter'), :id => 'submitButton') -%></p> + <%= link_to l(:button_apply), {}, :onclick => "$('submitButton').click(); return false;", :class => 'icon icon-checked' %> + <%= link_to l(:button_clear), {}, :class => 'icon icon-reload' %> + </div> + </fieldset> +<% end %> + +<div id="projects"> + <%= render :partial => 'filtered_projects' %> +</div> + + + + +<% other_formats_links do |f| %> + <%= f.link_to 'Atom', :url => {:key => User.current.rss_key} %> +<% end %> + +<% html_title(l(:label_project_plural)) -%>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/app/views/tags/_settings.html.erb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,14 @@ +<fieldset><legend><%= l(:setting_issue_tags) %></legend> + <p> + <label><%= l(:issues_sidebar) %></label> + <%= select_tag 'settings[issues_sidebar]', options_for_select(%w(none list cloud).collect{|v| [l("issue_tags_sidebar_#{v}"), v]}, @settings[:issues_sidebar]) %> + </p> + <p> + <label><%= l(:issues_show_count) %></label> + <%= check_box_tag 'settings[issues_show_count]', 1, 1 == @settings[:issues_show_count].to_i %> + </p> + <p> + <label><%= l(:issues_open_only) %></label> + <%= check_box_tag 'settings[issues_open_only]', 1, 1 == @settings[:issues_open_only].to_i %> + </p> +</fieldset>
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/assets/javascripts/projects_index.js Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,37 @@ +function toggleFieldsetWithState(this_field){ + id = Element.up(this_field, 'fieldset').id; + // is the fieldset collapsed? + status = $(id).hasClassName("collapsed"); + change_session(id, status); + + toggleFieldset(this_field); + +}; + + function submitForm(){ + $('submitButton').click(); + }; + +function change_session(id, nstatus) { + var url = "projects/set_fieldset_status"; + var request = new Ajax.Request(url, { + method: 'post', + parameters: {field_id: id, status: nstatus}, + asynchronous: true + }); +} + +function keypressHandler (event){ + var key = event.which || event.keyCode; + switch (key) { + default: + break; + case Event.KEY_RETURN: + $('submitButton').click(); return false; + break; + }; +}; + +document.observe("dom:loaded", function() { + $('search-input').observe('keypress', keypressHandler); +}); \ No newline at end of file
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/assets/javascripts/tags_input.js Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,121 @@ +/** + * This file is a part of redmine_tags + * redMine plugin, that adds tagging support. + * + * Copyright (c) 2010 Aleksey V Zapparov AKA ixti + * + * redmine_tags is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * redmine_tags is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + */ + +var Redmine = Redmine || {}; + +Redmine.TagsInput = Class.create({ + initialize: function(element, update) { + this.element = $(element); + this.input = new Element('input', { 'type': 'text', 'autocomplete': 'off', 'size': 10 }); + this.button = new Element('span', { 'class': 'tag-add icon icon-add' }); + this.tags = new Hash(); + + this.update = update; + + var uri_params = window.location.href.toQueryParams(); + if (uri_params["project[tag_list]"] != undefined){ + this.addTag(uri_params["project[tag_list]"].stripTags(), true); + }; + + Event.observe(this.button, 'click', this.readTags.bind(this)); + Event.observe(this.input, 'keypress', this.onKeyPress.bindAsEventListener(this)); + + this.element.insert({ 'after': this.input }); + this.input.insert({ 'after': this.button }); + this.addTagsList(this.element.value); + }, + + readTags: function() { + this.addTagsList(this.input.value); + this.input.value = ''; + if(this.update){ + submitForm(); + }; + }, + + onKeyPress: function(event) { + if (Event.KEY_RETURN == event.keyCode) { + this.readTags(event); + Event.stop(event); + } + }, + + addTag: function(tag, noSubmit) { + if (tag.blank() || this.tags.get(tag)) return; + + if(noSubmit==undefined){noSubmit=false;} + + var button = new Element('span', { 'class': 'tag-delete icon icon-del' }); + var label = new Element('span', { 'class': 'tag-label' }).insert(tag).insert(button); + + this.tags.set(tag, 1); + this.element.value = this.getTagsList(); + this.element.insert({ 'before': label }); + + if(noSubmit==false){ + if(this.update){ + submitForm(); + }; + }; + + Event.observe(button, 'click', function(){ + this.tags.unset(tag); + this.element.value = this.getTagsList(); + label.remove(); + if(this.update){submitForm();}; + }.bind(this)); + }, + + addTagsList: function(tags_list) { + var tags = tags_list.split(','); + for (var i = 0; i < tags.length; i++) { + this.addTag(tags[i].strip()); + } + }, + + getTagsList: function() { + return this.tags.keys().join(','); + }, + + autocomplete: function(container, url) { + new Ajax.Autocompleter(this.input, container, url, { + 'minChars': 1, + 'frequency': 0.5, + 'paramName': 'q', + 'updateElement': function(el) { + this.input.value = el.getAttribute('name'); + this.readTags(); + }.bind(this) + }); + } +}); + + +function observeIssueTagsField(url) { + new Redmine.TagsInput('issue_tag_list', false).autocomplete('issue_tag_candidates', url); +} + +function observeProjectTagsField(url, update) { + if(!update) { + var update = false; + }; + + new Redmine.TagsInput('project_tag_list', update).autocomplete('project_tag_candidates', url); +} \ No newline at end of file
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/assets/stylesheets/redmine_tags.css Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,35 @@ +/** + * This file is a part of redmine_tags + * redMine plugin, that adds tagging support. + * + * Copyright (c) 2010 Aleksey V Zapparov AKA ixti + * + * redmine_tags is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * redmine_tags is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + */ + +ul.tags { list-style: none; padding: 0px; } +ul.tags li { margin: .25em 0px; } + +div.tags { text-align: center; } +div.tags .tag-label { margin: .25em; } +div.tags .tag-nube-1 { font-size: .8em; } +div.tags .tag-nube-2 { font-size: .9em; } +div.tags .tag-nube-3 { font-size: 1em; } +div.tags .tag-nube-4 { font-size: 1.1em; } +div.tags .tag-nube-5 { font-size: 1.2em; } +div.tags .tag-nube-6 { font-size: 1.3em; } +div.tags .tag-nube-7 { font-size: 1.4em; } +div.tags .tag-nube-8 { font-size: 1.5em; } + +.tag-count { font-size: .75em; margin-left: .5em; }
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/config/locales/de.yml Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,35 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# German translation for redmine_tags +# by Terence Miller aka cforce, <cforce(at)gmx.de> +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +de: + tags: Tags + field_tags: Tags + field_tag_list: Tags + setting_issue_tags: Ticket Tags + issues_sidebar: Zeige die Tags auf der Sidebar + issues_show_count: Zeige die Ticketanzahl an + issues_open_only: Zeige nur noch offene Tickets + + issue_tags_sidebar_none: Keine + issue_tags_sidebar_list: Liste + issue_tags_sidebar_cloud: Cloud + + auto_complete_new_tag: Hinzufügen...
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/config/locales/en.yml Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,40 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# English translation for redmine_tags +# by Aleksey V Zapparov AKA ixti +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +en: + tags: Tags + field_tags: Tags + field_tag_list: Tags + label_tags_search: "Tags: " + setting_issue_tags: Issues Tags + issues_sidebar: Display tags on sidebar as + issues_show_count: Display amount of issues + issues_open_only: Display open issues only + + issue_tags_sidebar_none: None + issue_tags_sidebar_list: List + issue_tags_sidebar_cloud: Cloud + + auto_complete_new_tag: Add new... + + project_filtering_q_label: "Textual search" + project_filter_no_results: "No matching projects found" + button_filter: "Filter"
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/config/locales/fr.yml Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,35 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# French translation for redmine_tags +# by Stphane HANNEQUIN, <stephane.hannequin(at)aster-ingenierie.com> +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +fr: + tags: Tags + field_tags: Tags + field_tag_list: Tags + setting_issue_tags: Tags des demandes + issues_sidebar: Afficher les Tags comme + issues_show_count: Afficher le nombre de demande + issues_open_only: N'afficher que les demandes ouvertes + + issue_tags_sidebar_none: Ne pas afficher + issue_tags_sidebar_list: Liste + issue_tags_sidebar_cloud: Nuage + + auto_complete_new_tag: Nouveau...
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/config/locales/ru.yml Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,35 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Russian translation for redmine_tags +# by Aleksey V Zapparov AKA ixti +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +ru: + tags: Метки + field_tags: Метки + field_tag_list: Метки + setting_issue_tags: Метки задач + issues_sidebar: Боковую панель как + issues_show_count: Показать кол-во задач + issues_open_only: Только открытые задачи + + issue_tags_sidebar_none: Не показывать + issue_tags_sidebar_list: Список + issue_tags_sidebar_cloud: Облако + + auto_complete_new_tag: Добавить...
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/config/routes.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,3 @@ +ActionController::Routing::Routes.draw do |map| + map.connect 'projects/set_fieldset_status', :controller => 'projects', :action => 'set_fieldset_status', :conditions => {:method => :post} +end \ No newline at end of file
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/db/migrate/001_acts_as_taggable_on_migration.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,29 @@ +class ActsAsTaggableOnMigration < ActiveRecord::Migration + def self.up + create_table :tags do |t| + t.column :name, :string + end + + create_table :taggings do |t| + t.column :tag_id, :integer + t.column :taggable_id, :integer + t.column :tagger_id, :integer + t.column :tagger_type, :string + + # You should make sure that the column created is + # long enough to store the required class names. + t.column :taggable_type, :string + t.column :context, :string + + t.column :created_at, :datetime + end + + add_index :taggings, :tag_id + add_index :taggings, [:taggable_id, :taggable_type, :context] + end + + def self.down + drop_table :taggings + drop_table :tags + end +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/init.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,84 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +config.gem "acts-as-taggable-on", :version => '2.0.6' + +require 'redmine' + +Redmine::Plugin.register :redmine_tags do + name 'redmine_tags' + author 'Aleksey V Zapparov AKA "ixti"' + description 'redMine tagging support' + version '1.1.4' + url 'http://www.ixti.ru/' + author_url 'http://www.ixti.ru/' + + requires_redmine :version_or_higher => '1.0.0' + + settings :default => { + :issues_sidebar => 'none', + :issues_show_count => 0, + :issues_open_only => 0 + }, :partial => 'tags/settings' +end + + +require 'dispatcher' + +Dispatcher.to_prepare :redmine_tags do + + require_dependency 'redmine_project_filtering' + + unless Project.included_modules.include?(RedmineTags::Patches::ProjectPatch) + Project.send(:include, RedmineTags::Patches::ProjectPatch) + end + + unless ProjectsHelper.included_modules.include?(RedmineTags::Patches::ProjectsHelperPatch) + ProjectsHelper.send(:include, RedmineTags::Patches::ProjectsHelperPatch) + end + + unless Issue.included_modules.include?(RedmineTags::Patches::IssuePatch) + Issue.send(:include, RedmineTags::Patches::IssuePatch) + end + + unless IssuesHelper.included_modules.include?(RedmineTags::Patches::IssuesHelperPatch) + IssuesHelper.send(:include, RedmineTags::Patches::IssuesHelperPatch) + end + + unless ProjectsController.included_modules.include?(RedmineTags::Patches::ProjectsControllerPatch) + ProjectsController.send(:include, RedmineTags::Patches::ProjectsControllerPatch) + end + + unless AutoCompletesController.included_modules.include?(RedmineTags::Patches::AutoCompletesControllerPatch) + AutoCompletesController.send(:include, RedmineTags::Patches::AutoCompletesControllerPatch) + end + + unless Query.included_modules.include?(RedmineTags::Patches::QueryPatch) + Query.send(:include, RedmineTags::Patches::QueryPatch) + end + + unless QueriesHelper.included_modules.include?(RedmineTags::Patches::QueriesHelperPatch) + QueriesHelper.send(:include, RedmineTags::Patches::QueriesHelperPatch) + end +end + + +require 'redmine_tags/hooks/model_issue_hook' +require 'redmine_tags/hooks/views_issues_hook' +require 'redmine_tags/hooks/views_projects_hook' +
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_project_filtering.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,15 @@ +module RedmineProjectFiltering + + # transforms a question and a list of custom fields into something that Project.search can process + def self.calculate_tokens(question, custom_fields=nil) + list = [] + list << question if question.present? + + tokens = list.join(' ').scan(%r{((\s|^)"[\s\w]+"(\s|$)|\S+)}) + tokens = tokens.collect{ |m| m.first.gsub(%r{(^\s*"\s*|\s*"\s*$)}, '') } + + # tokens must be at least 2 characters long + tokens.select {|w| w.length > 1 } + end + +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/hooks/model_issue_hook.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,54 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Eric Davis +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +module RedmineTags + module Hooks + class ModelIssueHook < Redmine::Hook::ViewListener + def controller_issues_edit_before_save(context={}) + save_tags_to_issue(context, true) + end + + # Issue has an after_save method that calls reload (update_nested_set_attributes) + # This makes it impossible for a new record to get a tag_list, it's + # cleared on reload. So instead, hook in after the Issue#save to update + # this issue's tag_list and call #save ourselves. + def controller_issues_new_after_save(context={}) + save_tags_to_issue(context, false) + context[:issue].save + end + + def save_tags_to_issue(context, create_journal) + params = context[:params] + + if params && params[:issue] && !params[:issue][:tag_list].nil? + old_tags = context[:issue].tag_list.to_s + context[:issue].tag_list = params[:issue][:tag_list] + new_tags = context[:issue].tag_list.to_s + + if create_journal and not (old_tags == new_tags || context[:issue].current_journal.blank?) + context[:issue].current_journal.details << JournalDetail.new(:property => 'attr', + :prop_key => 'tag_list', + :old_value => old_tags, + :value => new_tags) + end + end + end + end + end +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/hooks/model_project_hook.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,58 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Eric Davis +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +module RedmineTags + module Hooks + class ModelProjectHook < Redmine::Hook::ViewListener + def controller_project_before_save(context={}) + debugger + save_tags_to_project(context, true) + end + + # Issue has an after_save method that calls reload (update_nested_set_attributes) + # This makes it impossible for a new record to get a tag_list, it's + # cleared on reload. So instead, hook in after the Issue#save to update + # this issue's tag_list and call #save ourselves. + def controller_projects_before_save(context={}) + debugger + save_tags_to_project(context, false) + context[:project].save + end + + def save_tags_to_project(context, create_journal) + params = context[:params] + debugger + logger.error { "WORKING" } + + # if params && params[:issue] && !params[:issue][:tag_list].nil? + # old_tags = context[:issue].tag_list.to_s + # context[:issue].tag_list = params[:issue][:tag_list] + # new_tags = context[:issue].tag_list.to_s + # + # if create_journal and not (old_tags == new_tags || context[:issue].current_journal.blank?) + # context[:issue].current_journal.details << JournalDetail.new(:property => 'attr', + # :prop_key => 'tag_list', + # :old_value => old_tags, + # :value => new_tags) + # end + # end + end + end + end +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/hooks/views_issues_hook.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,28 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +module RedmineTags + module Hooks + class ViewsIssuesHook < Redmine::Hook::ViewListener + render_on :view_issues_show_details_bottom, :partial => 'issues/tags' + render_on :view_issues_form_details_bottom, :partial => 'issues/tags_form' + render_on :view_issues_sidebar_planning_bottom, :partial => 'issues/tags_sidebar' + end + end +end +
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/hooks/views_projects_hook.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,10 @@ +module RedmineTags + module Hooks + class ViewsProjectsHook < Redmine::Hook::ViewListener + render_on :view_projects_form, :partial => 'projects/tags_form' + render_on :view_projects_show_left, :partial => 'projects/tags' +# render_on :view_issues_sidebar_planning_bottom, :partial => 'issues/tags_sidebar' + end + end +end +
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/patches/auto_completes_controller_patch.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,50 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +require_dependency 'auto_completes_controller' + +module RedmineTags + module Patches + module AutoCompletesControllerPatch + def self.included(base) + base.send(:include, InstanceMethods) + end + + + module InstanceMethods + def issue_tags + @name = params[:q].to_s + @tags = Issue.available_tags :project_id => @project, :name_like => @name + render :layout => false, :partial => 'tag_list' + end + + def project_tags + @name = params[:q].to_s + @tags = Project.available_tags :name_like => @name + render :layout => false, :partial => 'tag_list' + end + + def project_search_tags + @name = params[:q].to_s + @tags = Project.available_tags :name_like => @name + render :layout => false, :partial => 'search_tag_list' + end + end + end + end +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/patches/issue_patch.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,69 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +require_dependency 'issue' + +module RedmineTags + module Patches + module IssuePatch + def self.included(base) + base.extend(ClassMethods) + + base.class_eval do + unloadable + acts_as_taggable + end + end + + module ClassMethods + # Returns available issue tags + # === Parameters + # * <i>options</i> = (optional) Options hash of + # * project - Project to search in. + # * open_only - Boolean. Whenever search within open issues only. + # * name_like - String. Substring to filter found tags. + def available_tags(options = {}) + project = options[:project] + open_only = options[:open_only] + name_like = options[:name_like] + options = {} + visible = ARCondition.new + + if project + project = project.id if project.is_a? Project + visible << ["#{Issue.table_name}.project_id = ?", project] + end + + if open_only + visible << ["#{Issue.table_name}.status_id IN " + + "( SELECT issue_status.id " + + " FROM #{IssueStatus.table_name} issue_status " + + " WHERE issue_status.is_closed = ? )", false] + end + + if name_like + visible << ["#{ActsAsTaggableOn::Tag.table_name}.name LIKE ?", "%#{name_like.downcase}%"] + end + + options[:conditions] = visible.conditions + self.all_tag_counts(options) + end + end + end + end +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/patches/issues_helper_patch.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,56 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +require_dependency 'issues_helper' + +module RedmineTags + module Patches + module IssuesHelperPatch + def self.included(base) + base.send(:include, InstanceMethods) + end + + module InstanceMethods + include TagsHelper + + def redmine_tags_settings + @redmine_tags_settings = Setting.plugin_redmine_tags unless @redmine_tags_settings + @redmine_tags_settings + end + + def sidebar_tags + unless @sidebar_tags + @sidebar_tags = [] + if :none != redmine_tags_settings[:issues_sidebar].to_sym + @sidebar_tags = Issue.available_tags(:project => @project, + :open_only => (redmine_tags_settings[:issues_open_only].to_i == 1)) + end + end + @sidebar_tags + end + + def render_sidebar_tags + render_tags_list(sidebar_tags, + :show_count => (redmine_tags_settings[:issues_show_count].to_i == 1), + :open_only => (redmine_tags_settings[:issues_open_only].to_i == 1), + :style => redmine_tags_settings[:issues_sidebar].to_sym) + end + end + end + end +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/patches/project_patch.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,81 @@ +# C4DM + +require_dependency 'project' + +module RedmineTags + module Patches + module ProjectPatch + def self.included(base) # :nodoc: + base.extend(ClassMethods) + base.send(:include, InstanceMethods) + + base.class_eval do + unloadable + + attr_accessor :tag_list + + acts_as_taggable + + end + end + + def before_save_with_save_tags() +# debugger + logger.error { "GONNA SAVE TAG LIST" } + + +# params[:tag_list] + + + # logger.error { @project.name } + + # if params && params[:project] && !params[:project][:tag_list].nil? + # old_tags = context[:project].tag_list.to_s + # context[:project].tag_list = params[:project][:tag_list] + # new_tags = context[:project].tag_list.to_s + # + # unless (old_tags == new_tags || context[:project].current_journal.blank?) + # context[:project].current_journal.details << JournalDetail.new(:property => 'attr', + # :prop_key => 'tag_list', + # :old_value => old_tags, + # :value => new_tags) + # end + # end + end + + module InstanceMethods + + end + + module ClassMethods + def search_by_question(question) + if question.length > 1 + search(RedmineProjectFiltering.calculate_tokens(question), nil, :all_words => true).first.sort_by(&:lft) + else + all(:order => 'lft') + end + end + + + # Returns available project tags + # does not show tags from private projects + def available_tags( options = {} ) + + name_like = options[:name_like] + options = {} + visible = ARCondition.new + + visible << ["#{Project.table_name}.is_public = \"1\""] + + if name_like + visible << ["#{ActsAsTaggableOn::Tag.table_name}.name LIKE ?", "%#{name_like.downcase}%"] + end + + options[:conditions] = visible.conditions + + self.all_tag_counts(options) + end + end + end + end +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/patches/projects_controller_patch.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,143 @@ +require_dependency 'projects_controller' + +module RedmineTags + module Patches + module ProjectsControllerPatch + def self.included(base) + base.send(:include, InstanceMethods) + base.class_eval do + unloadable + skip_before_filter :authorize, :only => [:set_fieldset_status] + skip_before_filter :find_project, :only => [:set_fieldset_status] + before_filter :add_tags_to_project, :only => [:save, :update] +# before_filter :filter_projects, :only => :index + + alias :index filtered_index + end + end + + module InstanceMethods + + + + def add_tags_to_project + + if params && params[:project] && !params[:project][:tag_list].nil? + old_tags = @project.tag_list.to_s + new_tags = params[:project][:tag_list].to_s + + unless (old_tags == new_tags) + @project.tag_list = new_tags + end + end + end + + def paginate_projects + sort_init 'name' + sort_update %w(name lft created_on updated_on) + @limit = per_page_option + @project_count = Project.visible_roots.find(@projects).count + @project_pages = ActionController::Pagination::Paginator.new self, @project_count, @limit, params['page'] + @offset ||= @project_pages.current.offset + end + + def set_fieldset_status + + # luisf. test for missing parameters……… + field = params[:field_id] + status = params[:status] + + session[(field + "_status").to_sym] = status + render :nothing => true + end + + # gets the status of the collabsible fieldsets + def get_fieldset_statuses + if session[:my_projects_fieldset_status].nil? + @myproj_status = "true" + else + @myproj_status = session[:my_projects_fieldset_status] + end + + if session[:filters_fieldset_status].nil? + @filter_status = "false" + else + @filter_status = session[:filters_fieldset_status] + end + + if params && params[:project] && !params[:project][:tag_list].nil? + @filter_status = "true" + end + + end + + # Lists visible projects. Paginator is for top-level projects only + # (subprojects belong to them) + def filtered_index + @project = Project.new + filter_projects + get_fieldset_statuses + + respond_to do |format| + format.html { + paginate_projects + + @projects = Project.visible_roots.find(@projects, :offset => @offset, :limit => @limit, :order => sort_clause) + + if User.current.logged? + # seems sort_by gives us case-sensitive ordering, which we don't want + # @user_projects = User.current.projects.sort_by(&:name) + @user_projects = User.current.projects.all(:order => :name) + end + + render :template => 'projects/index.rhtml', :layout => !request.xhr? + } + format.api { + @offset, @limit = api_offset_and_limit + @project_count = Project.visible.count + @projects = Project.visible.find(@projects, :offset => @offset, :limit => @limit, :order => 'lft') + } + format.atom { + projects = Project.visible.find(:all, :order => 'created_on DESC', :limit => Setting.feeds_limit.to_i) + render_feed(projects, :title => "#{Setting.app_title}: #{l(:label_project_latest)}") + } + format.js { + paginate_projects + @projects = Project.visible_roots.find(@projects, :offset => @offset, :limit => @limit, :order => sort_clause) + render :update do |page| + page.replace_html 'projects', :partial => 'filtered_projects' + end + } + end + end + + private + + def filter_projects + @question = (params[:q] || "").strip + + if params.has_key?(:project) + @tag_list = (params[:project][:tag_list] || "").strip.split(",") + else + @tag_list = [] + end + + if @question == "" + @projects = Project.visible + else + @projects = Project.visible.search_by_question(@question) + end + + unless @tag_list.empty? + @tagged_projects_ids = Project.visible.tagged_with(@tag_list).collect{ |project| Project.find(project.id) } + @projects = @projects & @tagged_projects_ids + end + + @projects = @projects.collect{ |project| project.root } + @projects = @projects.uniq + + end + end + end + end +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/patches/projects_helper_patch.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,262 @@ +module RedmineTags + module Patches + module ProjectsHelperPatch + + def self.included(base) # :nodoc: + base.send(:include, InstanceMethods) + base.class_eval do + unloadable + end + end + + module InstanceMethods + # Renders a tree of projects that the current user does not belong + # to, or of all projects if the current user is not logged in. The + # given collection may be a subset of the whole project tree + # (eg. some intermediate nodes are private and can not be seen). We + # are potentially interested in various things: the project name, + # description, manager(s), creation date, last activity date, + # general activity level, whether there is anything actually hosted + # here for the project, etc. + def render_project_table_with_filtering(projects, question) + custom_fields = "" + s = "" + if projects.any? + tokens = RedmineProjectFiltering.calculate_tokens(question, custom_fields) + + s << "<div class='autoscroll'>" + s << "<table class='list projects'>" + s << "<thead><tr>" + + s << sort_header_tag('name', :caption => l("field_name")) + s << "<th class='managers'>" << l("label_managers") << "</th>" + s << "<th class='tags'>" << l("tags") << "</th>" + s << sort_header_tag('created_on', :default_order => 'desc') + s << sort_header_tag('updated_on', :default_order => 'desc') + + s << "</tr></thead><tbody>" + + original_project = @project + + projects.each do |project| + s << render_project_in_table_with_filtering(project, cycle('odd', 'even'), 0, tokens) + end + + s << "</table>" + else + s << "\n" + end + @project = original_project + + s + end + + def render_project_in_table_with_filtering(project, oddeven, level, tokens) + # set the project environment to please macros. + @project = project + + classes = (level == 0 ? 'root' : 'child') + + s = "" + + s << "<tr class='#{oddeven} #{classes} level#{level}'>" + s << "<td class='firstcol' align=top><div class='name hosted_here" + s << " no_description" if project.description.blank? + s << "'>" << link_to( highlight_tokens(project.name, tokens), {:controller => 'projects', :action => 'show', :id => project}, :class => "project #{User.current.member_of?(project) ? 'my-project' : nil}") + s << "</div>" + s << highlight_tokens(render_project_short_description(project), tokens) + s << "<td class='managers' align=top>" + + u = project.users_by_role + if u + u.keys.each do |r| + if r.allowed_to?(:edit_project) + mgrs = [] + u[r].sort.each do |m| + mgrs << link_to_user(m) + end + if mgrs.size < 3 + s << '<nobr>' << mgrs.join(', ') << '</nobr>' + else + s << mgrs.join(', ') + end + end + end + end + + s << "</td>" + + # taglist + s << "<td class='tags' align=top>" << project.tag_counts.collect{ |t| render_project_tag_link(t) }.join(', ') << "</td>" + s << "<td class='created_on' align=top>" << format_date(project.created_on) << "</td>" + s << "<td class='updated_on' align=top>" << format_date(project.updated_on) << "</td>" + + s << "</tr>" + + project.children.each do |child| + if child.is_public? or User.current.member_of?(child) + s << render_project_in_table_with_filtering(child, oddeven, level + 1, tokens) + end + end + + s + end + + + + # Renders a tree of projects as a nested set of unordered lists + # The given collection may be a subset of the whole project tree + # (eg. some intermediate nodes are private and can not be seen) + def render_project_hierarchy_with_filtering(projects,custom_fields,question) + s = [] + if projects.any? + tokens = RedmineProjectFiltering.calculate_tokens(question, custom_fields) + debugger + + + ancestors = [] + original_project = @project + projects.each do |project| + # set the project environment to please macros. + @project = project + if (ancestors.empty? || project.is_descendant_of?(ancestors.last)) + s << "<ul class='projects #{ ancestors.empty? ? 'root' : nil}'>" + else + ancestors.pop + s << "</li>" + while (ancestors.any? && !project.is_descendant_of?(ancestors.last)) + ancestors.pop + s << "</ul></li>" + end + end + classes = (ancestors.empty? ? 'root' : 'child') + s << "<li class='#{classes}'><div class='#{classes}'>" + + link_to( highlight_tokens(project.name, tokens), + {:controller => 'projects', :action => 'show', :id => project}, + :class => "project #{User.current.member_of?(project) ? 'my-project' : nil}" + ) + s << "<ul class='filter_fields'>" + + # CustomField.usable_for_project_filtering.each do |field| + # value_model = project.custom_value_for(field.id) + # value = value_model.present? ? value_model.value : nil + # s << "<li><b>#{field.name.humanize}:</b> #{highlight_tokens(value, tokens)}</li>" if value.present? + # end + + s << "</ul>" + s << "<div class='clear'></div>" + unless project.description.blank? + s << "<div class='wiki description'>" + s << "<b>#{ t(:field_description) }:</b>" + s << highlight_tokens(textilizable(project.short_description, :project => project), tokens) + s << "\n</div>" + end + s << "</div>" + ancestors << project + end + ancestors.size.times{ s << "</li></ul>" } + @project = original_project + end + s.join "\n" + end + + # Renders a tree of projects where the current user belongs + # as a nested set of unordered lists + # The given collection may be a subset of the whole project tree + # (eg. some intermediate nodes are private and can not be seen) + def render_my_project_hierarchy_with_tags(projects) + + s = '' + + original_project = @project + + projects.each do |project| + if project.root? || !projects.include?(project.parent) + s << render_my_project_in_hierarchy_with_tags(project) + end + end + + @project = original_project + + if s != '' + a = '' + a << "<ul class='projects root'>\n" + a << s + a << "</ul>\n" + s = a + end + + s + + end + + + + + def render_my_project_in_hierarchy_with_tags(project) + + s = '' + + if User.current.member_of?(project) + + # set the project environment to please macros. + @project = project + + classes = (project.root? ? 'root' : 'child') + + s << "<li class='#{classes}'><div class='#{classes}'>" + + link_to_project(project, {}, :class => "project my-project") + if project.is_public? + s << " <span class='public'>" << l("field_is_public") << "</span>" + else + s << " <span class='private'>" << l("field_is_private") << "</span>" + end + s << render_project_short_description(project) + + s << l(:tags) << ": " + s << project.tag_counts.collect{ |t| render_project_tag_link(t) }.join(', ') + + s << "</div>\n" + + cs = '' + project.children.each do |child| + cs << render_my_project_in_hierarchy(child) + end + + if cs != '' + s << "<ul class='projects'>\n" << cs << "</ul>\n"; + end + + end + + s + + end + + + + private + + # copied from search_helper. This one doesn't escape html or limit the text length + def highlight_tokens(text, tokens) + return text unless text && tokens && !tokens.empty? + re_tokens = tokens.collect {|t| Regexp.escape(t)} + regexp = Regexp.new "(#{re_tokens.join('|')})", Regexp::IGNORECASE + result = '' + text.split(regexp).each_with_index do |words, i| + words = words.mb_chars + if i.even? + result << words + else + t = (tokens.index(words.downcase) || 0) % 4 + result << content_tag('span', words, :class => "highlight token-#{t}") + end + end + result + end + + end + end + end +end +
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/patches/queries_helper_patch.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,48 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +require_dependency 'queries_helper' + +module RedmineTags + module Patches + module QueriesHelperPatch + def self.included(base) + base.send(:include, InstanceMethods) + + base.class_eval do + alias_method :column_content_original, :column_content + alias_method :column_content, :column_content_extended + end + end + + + module InstanceMethods + include TagsHelper + + + def column_content_extended(column, issue) + if column.name.eql? :tags + column.value(issue).collect{ |t| render_tag_link(t) }.join(', ') + else + column_content_original(column, issue) + end + end + end + end + end +end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/vendor/plugins/redmine_tags/lib/redmine_tags/patches/query_patch.rb Wed Nov 23 11:15:54 2011 +0000 @@ -0,0 +1,74 @@ +# This file is a part of redmine_tags +# redMine plugin, that adds tagging support. +# +# Copyright (c) 2010 Aleksey V Zapparov AKA ixti +# +# redmine_tags is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# redmine_tags is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with redmine_tags. If not, see <http://www.gnu.org/licenses/>. + +require_dependency 'query' + +module RedmineTags + module Patches + module QueryPatch + def self.included(base) + base.send(:include, InstanceMethods) + + base.class_eval do + unloadable + + alias_method :statement_original, :statement + alias_method :statement, :statement_extended + + alias_method :available_filters_original, :available_filters + alias_method :available_filters, :available_filters_extended + + base.add_available_column(QueryColumn.new(:tags)) + end + end + + + module InstanceMethods + def statement_extended + filter = filters.delete 'tags' + clauses = statement_original + + if filter + filters.merge!( 'tags' => filter ) + + values = values_for('tags').clone + compare = operator_for('tags').eql?('=') ? 'IN' : 'NOT IN' + ids_list = Issue.tagged_with(values).collect{ |issue| issue.id }.push(0).join(',') + + clauses << " AND ( #{Issue.table_name}.id #{compare} (#{ids_list}) ) " + end + + clauses + end + + + def available_filters_extended + unless @available_filters + available_filters_original.merge!({ 'tags' => { + :type => :list, + :order => 6, + :values => Issue.available_tags(:project => project).collect{ |t| [t.name, t.name] } + }}) + end + @available_filters + end + end + end + end +end +