diff options
author | Go MAEDA <maeda@farend.jp> | 2021-03-18 03:58:29 +0000 |
---|---|---|
committer | Go MAEDA <maeda@farend.jp> | 2021-03-18 03:58:29 +0000 |
commit | 44ea826a21071ab3e0b07ecfcd56020eea219078 (patch) | |
tree | 6006b28121b01d573d3545c23010bea66e8c189e | |
parent | 06ac3473e0eb9d233611320ca2ed6d2c3e4f6547 (diff) | |
download | redmine-44ea826a21071ab3e0b07ecfcd56020eea219078.tar.gz redmine-44ea826a21071ab3e0b07ecfcd56020eea219078.zip |
"Copy link" feature for issue and issue journal (#34703).
Patch by Mizuki ISHIKAWA.
git-svn-id: http://svn.redmine.org/redmine/trunk@20816 e93f8b46-1217-0410-a6f0-8f06a7374b81
-rw-r--r-- | app/helpers/application_helper.rb | 8 | ||||
-rw-r--r-- | app/helpers/journals_helper.rb | 4 | ||||
-rw-r--r-- | app/views/issues/_action_menu.html.erb | 1 | ||||
-rw-r--r-- | config/locales/en.yml | 1 | ||||
-rw-r--r-- | public/images/copy_link.png | bin | 0 -> 1731 bytes | |||
-rw-r--r-- | public/javascripts/application.js | 17 | ||||
-rw-r--r-- | public/stylesheets/application.css | 1 | ||||
-rw-r--r-- | test/functional/issues_controller_test.rb | 26 | ||||
-rw-r--r-- | test/helpers/journals_helper_test.rb | 1 | ||||
-rw-r--r-- | test/system/copy_to_clipboard_test.rb | 63 |
10 files changed, 109 insertions, 13 deletions
diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb index 5163ea181..fe95eea47 100644 --- a/app/helpers/application_helper.rb +++ b/app/helpers/application_helper.rb @@ -1818,6 +1818,14 @@ module ApplicationHelper ) end + def copy_object_url_link(url) + link_to_function( + l(:button_copy_link), 'copyTextToClipboard(this);', + class: 'icon icon-copy-link', + data: {'clipboard-text' => url} + ) + end + private def wiki_helper diff --git a/app/helpers/journals_helper.rb b/app/helpers/journals_helper.rb index 4c3d410ec..ef649a278 100644 --- a/app/helpers/journals_helper.rb +++ b/app/helpers/journals_helper.rb @@ -29,9 +29,11 @@ module JournalsHelper def render_journal_actions(issue, journal, options={}) links = [] dropbown_links = [] + indice = journal.indice || @journal.issue.visible_journals_with_index.find{|j| j.id == @journal.id}.indice + + dropbown_links << copy_object_url_link(issue_url(issue, anchor: "note-#{indice}", only_path: false)) if journal.notes.present? if options[:reply_links] - indice = journal.indice || @journal.issue.visible_journals_with_index.find{|j| j.id == @journal.id}.indice links << link_to(l(:button_quote), quoted_issue_path(issue, :journal_id => journal, :journal_indice => indice), :remote => true, diff --git a/app/views/issues/_action_menu.html.erb b/app/views/issues/_action_menu.html.erb index 93e5bad63..a1dcd6312 100644 --- a/app/views/issues/_action_menu.html.erb +++ b/app/views/issues/_action_menu.html.erb @@ -6,6 +6,7 @@ :class => 'icon icon-time-add' if User.current.allowed_to?(:log_time, @project) %> <%= watcher_link(@issue, User.current) %> <%= actions_dropdown do %> + <%= copy_object_url_link(issue_url(@issue, only_path: false)) %> <%= link_to l(:button_copy), project_copy_issue_path(@project, @issue), :class => 'icon icon-copy' if User.current.allowed_to?(:copy_issues, @project) && Issue.allowed_target_projects.any? %> <%= link_to l(:button_delete), issue_path(@issue), diff --git a/config/locales/en.yml b/config/locales/en.yml index 5cbb33ffc..2ff0ae432 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -1149,6 +1149,7 @@ en: button_change_password: Change password button_copy: Copy button_copy_and_follow: Copy and follow + button_copy_link: Copy link button_annotate: Annotate button_fetch_changesets: Fetch commits button_update: Update diff --git a/public/images/copy_link.png b/public/images/copy_link.png Binary files differnew file mode 100644 index 000000000..2a2eb09ff --- /dev/null +++ b/public/images/copy_link.png diff --git a/public/javascripts/application.js b/public/javascripts/application.js index 770d5900c..bef8f592f 100644 --- a/public/javascripts/application.js +++ b/public/javascripts/application.js @@ -563,6 +563,23 @@ function randomKey(size) { return key; } +function copyTextToClipboard(target) { + if (target) { + var temp = document.createElement('textarea'); + temp.value = target.getAttribute('data-clipboard-text'); + document.body.appendChild(temp); + temp.select(); + document.execCommand('copy'); + if (temp.parentNode) { + temp.parentNode.removeChild(temp); + } + if ($(target).closest('.drdn.expanded').length) { + $(target).closest('.drdn.expanded').removeClass("expanded"); + } + } + return false; +} + function updateIssueFrom(url, el) { $('#all_attributes input, #all_attributes textarea, #all_attributes select').each(function(){ $(this).data('valuebeforeupdate', $(this).val()); diff --git a/public/stylesheets/application.css b/public/stylesheets/application.css index 1c2f76c15..0c4102459 100644 --- a/public/stylesheets/application.css +++ b/public/stylesheets/application.css @@ -1612,6 +1612,7 @@ td.gantt_selected_column .gantt_hdr,.gantt_selected_column_container { .icon-file.application-pdf { background-image: url(../images/files/pdf.png); } .icon-file.application-zip { background-image: url(../images/files/zip.png); } .icon-file.application-gzip { background-image: url(../images/files/zip.png); } +.icon-copy-link { background-image: url(../images/copy_link.png); } .sort-handle.ajax-loading { background-image: url(../images/loading.gif); } tr.ui-sortable-helper { border:1px solid #e4e4e4; } diff --git a/test/functional/issues_controller_test.rb b/test/functional/issues_controller_test.rb index 45217b696..dd85bb5d1 100644 --- a/test/functional/issues_controller_test.rb +++ b/test/functional/issues_controller_test.rb @@ -2072,12 +2072,13 @@ class IssuesControllerTest < Redmine::ControllerTest get(:show, :params => {:id => 1}) assert_response :success assert_select 'div.issue div.description', :text => /Unable to print recipes/ - assert_select '.contextual' do - assert_select 'a', {:count => 2, :text => /Edit/} - assert_select 'a', {:count => 0, :text => /Log time/} - assert_select 'a', {:count => 0, :text => /Watch/} - assert_select 'div.drdn-items a', {:count => 0, :text => /Copy/} - assert_select 'div.drdn-items a', {:count => 0, :text => /Delete/} + assert_select '#content>.contextual:first-child' do + assert_select 'a', {:count => 1, :text => 'Edit'} + assert_select 'a', {:count => 0, :text => 'Log time'} + assert_select 'a', {:count => 0, :text => 'Watch'} + assert_select 'div.drdn-items a', {:count => 1, :text => 'Copy link'} + assert_select 'div.drdn-items a', {:count => 0, :text => 'Copy'} + assert_select 'div.drdn-items a', {:count => 0, :text => 'Delete'} end # anonymous role is allowed to add a note assert_select 'form#issue-form' do @@ -2093,12 +2094,13 @@ class IssuesControllerTest < Redmine::ControllerTest @request.session[:user_id] = 2 get(:show, :params => {:id => 1}) assert_select 'a', :text => /Quote/ - assert_select '.contextual' do - assert_select 'a', {:count => 2, :text => /Edit/} - assert_select 'a', :text => /Log time/ - assert_select 'a', :text => /Watch/ - assert_select 'div.drdn-items a', :text => /Copy/ - assert_select 'div.drdn-items a', :text => /Delete/ + assert_select '#content>.contextual:first-child' do + assert_select 'a', {:count => 1, :text => 'Edit'} + assert_select 'a', {:count => 1, :text => 'Log time'} + assert_select 'a', {:count => 1, :text => 'Watch'} + assert_select 'div.drdn-items a', {:count => 1, :text => 'Copy link'} + assert_select 'div.drdn-items a', {:count => 1, :text => 'Copy'} + assert_select 'div.drdn-items a', {:count => 1, :text => 'Delete'} end assert_select 'form#issue-form' do assert_select 'fieldset' do diff --git a/test/helpers/journals_helper_test.rb b/test/helpers/journals_helper_test.rb index a75992c41..69628b0b5 100644 --- a/test/helpers/journals_helper_test.rb +++ b/test/helpers/journals_helper_test.rb @@ -59,6 +59,7 @@ class JournalsHelperTest < Redmine::HelperTest assert_select_in journal_actions, 'a[title=?][class="icon-only icon-comment"]', 'Quote' assert_select_in journal_actions, 'a[title=?][class="icon-only icon-edit"]', 'Edit' assert_select_in journal_actions, 'div[class="drdn-items"] a[class="icon icon-del"]' + assert_select_in journal_actions, 'div[class="drdn-items"] a[class="icon icon-copy-link"]' end def test_journal_thumbnail_attachments_should_be_in_the_same_order_as_the_journal_details diff --git a/test/system/copy_to_clipboard_test.rb b/test/system/copy_to_clipboard_test.rb new file mode 100644 index 000000000..ffa7b79fe --- /dev/null +++ b/test/system/copy_to_clipboard_test.rb @@ -0,0 +1,63 @@ +# frozen_string_literal: true + +# Redmine - project management software +# Copyright (C) 2006-2020 Jean-Philippe Lang +# +# This program 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 2 +# of the License, or (at your option) any later version. +# +# This program 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 this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + +require_relative '../application_system_test_case' + +class CopyToClipboardSystemTest < ApplicationSystemTestCase + fixtures :projects, :users, :email_addresses, :roles, :members, :member_roles, + :trackers, :projects_trackers, :enabled_modules, :issue_statuses, :issues, + :enumerations, :custom_fields, :custom_values, :custom_fields_trackers, + :watchers, :journals, :journal_details, :versions, + :workflows, :wikis, :wiki_pages, :wiki_contents, :wiki_content_versions + + def test_copy_issue_url_to_clipboard + log_user('jsmith', 'jsmith') + visit 'issues/1' + + # Copy issue url to Clipboard + first('.contextual span.icon-actions').click + find('.contextual div.drdn-items a.icon-copy-link').click + + # Paste the value copied to the clipboard into the textarea to get and test + first('.icon-edit').click + find('textarea#issue_notes').send_keys([modifier_key, 'v']) + assert find('textarea#issue_notes').value.end_with?('/issues/1') + end + + def test_copy_issue_journal_url_to_clipboard + log_user('jsmith', 'jsmith') + visit 'issues/1' + + # Copy issue journal url to Clipboard + first('#note-2 .icon-actions').click + first('#note-2 div.drdn-items a.icon-copy-link').click + + # Paste the value copied to the clipboard into the textarea to get and test + first('.icon-edit').click + find('textarea#issue_notes').send_keys([modifier_key, 'v']) + assert find('textarea#issue_notes').value.end_with?('/issues/1#note-2') + end + + private + + def modifier_key + modifier = osx? ? 'command' : 'control' + modifier.to_sym + end +end |