Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

497 edit page related external links #2490

Open
wants to merge 27 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
6516d52
Create basic UI
davidtrussler Dec 3, 2024
e4f96f5
Prepopulate form on initial load
davidtrussler Dec 6, 2024
2051914
Update view for multiple fields display
davidtrussler Dec 6, 2024
a3e932d
Update controller to add `update_related_external_links` action
davidtrussler Dec 6, 2024
e990e40
Update routes for `update_related_external_links` action
davidtrussler Dec 6, 2024
d848b72
Add checkbox partial
davidtrussler Dec 6, 2024
5e6f89c
Add hidden id to each form set
davidtrussler Dec 13, 2024
91a8762
Only render hidden field if id is sent
davidtrussler Dec 13, 2024
5654ef4
Update controller test for `update_related_external_links` method (WIP)
davidtrussler Dec 16, 2024
9eb0701
Update view to show empty fields where no external links exist
davidtrussler Dec 16, 2024
3074cc7
Update `edition_edit` test for `related external links` tab (WIP)
davidtrussler Dec 16, 2024
da8e5f3
Update view for rendering empty field
davidtrussler Dec 16, 2024
5d9baf6
Update integration test for empty form
davidtrussler Dec 16, 2024
152cc15
Update integration test for pre-populated form (WIP)
davidtrussler Dec 17, 2024
eb4c58d
Update controller and test
davidtrussler Jan 6, 2025
8fc0d20
Tidy up integration test
davidtrussler Jan 6, 2025
2013220
Update UI
davidtrussler Jan 6, 2025
28ecf3c
Update controller test
davidtrussler Jan 7, 2025
4e7e002
Update integration test for non-JS scenarios
davidtrussler Jan 7, 2025
61b158b
Update integration test for JS scenarios
davidtrussler Jan 8, 2025
ce070ed
Add controller test for scenario where there are no external links to…
davidtrussler Jan 10, 2025
0ed49c8
Rename `linking` to `related_external_links`
davidtrussler Jan 10, 2025
8b4b482
Update controller test to remove unecesary `artefact.reload`
davidtrussler Jan 10, 2025
e269831
Update controller test for typo
davidtrussler Jan 10, 2025
7e210a3
Remove JS tests from integration test
davidtrussler Jan 10, 2025
4baa904
Add new integration test for JS scenario
davidtrussler Jan 10, 2025
36b21e9
Revert helper to remove JS test methods
davidtrussler Jan 10, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions app/assets/javascripts/application.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,5 +2,6 @@

//= require govuk_publishing_components/dependencies
//= require govuk_publishing_components/lib
//= require govuk_publishing_components/components/add-another
//= require govuk_publishing_components/components/govspeak
//= require govuk_publishing_components/components/table
1 change: 1 addition & 0 deletions app/assets/stylesheets/application.scss
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ $govuk-page-width: 1140px;
@import 'govuk_publishing_components/govuk_frontend_support';
@import 'govuk_publishing_components/component_support';
@import 'govuk_publishing_components/components/button';
@import 'govuk_publishing_components/components/add-another';
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Very minor, but could we move this up a line to retain alphabetical ordering?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That would make sense but I found that some styles for the gem-c-add-another__remove-button class get over-written by the generic button styles when they are in that order. That seems to explain the spacing issue you have noted elsewhere WRT the delete button as well. It does seem a little bit unsatisfactory though so I wonder if we should make a change to the component stylesheet to fix this because I should think it's bound to happen again somewhere else.

@import 'govuk_publishing_components/components/checkboxes';
@import 'govuk_publishing_components/components/date-input';
@import 'govuk_publishing_components/components/details';
Expand Down
24 changes: 23 additions & 1 deletion app/controllers/editions_controller.rb
Original file line number Diff line number Diff line change
Expand Up @@ -79,10 +79,28 @@ def history
render action: "show"
end

def linking
def related_external_links
render action: "show"
end

def update_related_external_links
artefact = resource.artefact

if params.key?("artefact")
artefact.assign_attributes(permitted_external_links_params)

if artefact.save
flash[:success] = "Related links updated."
else
flash[:danger] = artefact.errors.full_messages.join("\n")
end
else
flash[:danger] = "There aren't any external related links yet"
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I can't find a test for this branch?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, yes, that's a good point. I've added a test but it might need more thinking about because that scenario never occurs with the new interface because it always presents the user with fields even if they are left blank (there's no equivalent to this in the legacy version). Maybe that whole branch isn't really necessary any more. This also relates to the problem of validating the two blank fields which currently doesn't happen but returns a success message even though nothing has changed. 🤔

end

redirect_to related_external_links_edition_path(@resource.id)
end

def confirm_unpublish
if redirect_url.blank? || validate_redirect(redirect_url)
render "secondary_nav_tabs/confirm_unpublish"
Expand Down Expand Up @@ -217,6 +235,10 @@ def permitted_update_params
params.require(:edition).permit(%i[title overview in_beta body major_change change_note])
end

def permitted_external_links_params
params.require(:artefact).permit(external_links_attributes: %i[title url id _destroy])
end

def require_destroyable
return if @resource.can_destroy?

Expand Down
2 changes: 2 additions & 0 deletions app/helpers/tabbed_nav_helper.rb
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,8 @@ def current_tab_name
"unpublish"
when "admin"
"admin"
when "related_external_links"
"related_external_links"
else
"temp_nav_text"
end
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
<%= render "govuk_publishing_components/components/checkboxes", {
name: "artefact[external_links_attributes][#{index}][_destroy]",
id: "artefact_external_links_attributes_#{index}__destroy",
items: [{label: "Delete", value: "1" }],
} %>
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
<% if id %>
<%= hidden_field_tag "artefact[external_links_attributes][#{index}][id]", id %>
<% end %>

<%= render "govuk_publishing_components/components/input", {
label: {
text: "Title",
bold: true,
},
name: "artefact[external_links_attributes][#{index}][title]",
id: "artefact_external_links_attributes_#{index}_title",
value: @edition.artefact.external_links[index].present? ? @edition.artefact.external_links[index].title : "",
} %>

<%= render "govuk_publishing_components/components/input", {
label: {
text: "URL",
bold: true,
},
name: "artefact[external_links_attributes][#{index}][url]",
id: "artefact_external_links_attributes_#{index}_url",
value: @edition.artefact.external_links[index].present? ? @edition.artefact.external_links[index].url : "",
hint: "Example: https://gov.uk",
} %>
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
<div class="govuk-grid-row">
<div class="govuk-grid-column-two-thirds">
<%= header_for("Related external links") %>

<%= form_for @edition.artefact, url: update_related_external_links_edition_path(@resource) do %>
<%
if @edition.artefact.external_links.count == 0
items = [{
fields: render(partial: "secondary_nav_tabs/add-another_fieldset", locals: { index: 0, id: nil }),
destroy_checkbox: render(partial: "secondary_nav_tabs/add-another_checkbox", locals: {index: 0}),
}]
empty = render(partial: "secondary_nav_tabs/add-another_fieldset", locals: { index: 1, id: nil })
else
items = @edition.artefact.external_links.each_with_index.map do | external_link, index |
{
fields: render(partial: "secondary_nav_tabs/add-another_fieldset", locals: { index:, id: external_link.id }),
destroy_checkbox: render(partial: "secondary_nav_tabs/add-another_checkbox", locals: {index: index}),
}
end
empty = render(partial: "secondary_nav_tabs/add-another_fieldset", locals: { index: @edition.artefact.external_links.count, id: nil })
end
%>

<%= render "govuk_publishing_components/components/add_another", {
fieldset_legend: "Link",
add_button_text: "Add another link",
items: items,
empty: empty,
} %>

<%= render "govuk_publishing_components/components/inset_text", {
text: "After saving, changes to related external links will be visible on the site the next time this publication is published.",
} %>

<%= render "govuk_publishing_components/components/button", {
text: "Save",
} %>
<% end %>
</div>
</div>
3 changes: 2 additions & 1 deletion config/routes.rb
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,8 @@
get "history"
get "admin"
post "duplicate"
get "related_external_links", to: "editions#linking"
get "related_external_links"
patch "update_related_external_links"
get "tagging", to: "editions#linking"
get "unpublish"
get "unpublish/confirm-unpublish", to: "editions#confirm_unpublish", as: "confirm_unpublish"
Expand Down
60 changes: 58 additions & 2 deletions test/functional/editions_controller_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ class EditionsControllerTest < ActionController::TestCase
end

context "when 'restrict_access_by_org' feature toggle is disabled" do
%i[show metadata history admin linking unpublish].each do |action|
%i[show metadata history admin related_external_links unpublish].each do |action|
context "##{action}" do
setup do
@edition = FactoryBot.create(:edition, owning_org_content_ids: %w[org-two])
Expand Down Expand Up @@ -74,7 +74,7 @@ class EditionsControllerTest < ActionController::TestCase
test_strategy.switch!(:restrict_access_by_org, false)
end

%i[show metadata history admin linking unpublish].each do |action|
%i[show metadata history admin related_external_links unpublish].each do |action|
context "##{action}" do
setup do
@edition = FactoryBot.create(:edition, owning_org_content_ids: %w[org-two])
Expand Down Expand Up @@ -625,6 +625,62 @@ class EditionsControllerTest < ActionController::TestCase
end
end

context "#update_related_external_links" do
should "display an error message when the title is blank" do
patch :update_related_external_links, params: {
id: @edition.id,
artefact: {
external_links_attributes: [{ title: "", url: "http://foo-bar.com", _destroy: false }],
},
}

assert_equal "External links is invalid", flash[:danger]
end

should "display an error message when the url is blank" do
patch :update_related_external_links, params: {
id: @edition.id,
artefact: {
external_links_attributes: [{ title: "foo", url: "", _destroy: false }],
},
}

assert_equal "External links is invalid", flash[:danger]
end

should "display an error message when the url is invalid" do
patch :update_related_external_links, params: {
id: @edition.id,
artefact: {
external_links_attributes: [{ title: "foo", url: "an-invalid-url", _destroy: false }],
},
}

assert_equal "External links is invalid", flash[:danger]
end

should "update related external links and display a success message when successfully saved" do
patch :update_related_external_links, params: {
id: @edition.id,
artefact: {
external_links_attributes: [{ title: "foo", url: "https://foo-bar.com", _destroy: false }],
},
}

assert_equal "Related links updated.", flash[:success]
assert_equal "foo", @edition.artefact.external_links[0].title
assert_equal "https://foo-bar.com", @edition.artefact.external_links[0].url
end

should "display an error message when there are no external links to save" do
patch :update_related_external_links, params: {
id: @edition.id,
}

assert_equal "There aren't any external related links yet", flash[:danger]
end
end

private

def description(edition)
Expand Down
124 changes: 124 additions & 0 deletions test/integration/edition_edit_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -585,6 +585,130 @@ class EditionEditTest < IntegrationTest
end
end

context "Related external links tab" do
setup do
visit_draft_edition
click_link "Related external links"
end

should "render 'Related external links' header, inset text and save button" do
assert page.has_css?("h2", text: "Related external links")
assert page.has_css?("div.gem-c-inset-text", text: "After saving, changes to related external links will be visible on the site the next time this publication is published.")
assert page.has_css?("button.gem-c-button", text: "Save")
end

context "Document has no external links when page loads" do
setup do
visit_draft_edition
click_link "Related external links"
end

should "render an empty 'Add another' form" do
# Link 1
assert page.has_css?("legend", text: "Link 1")
assert page.has_css?("input[name='artefact[external_links_attributes][0][_destroy]']")
assert_equal "Title", page.find("label[for='artefact_external_links_attributes_0_title']").text
assert_equal "URL", page.find("label[for='artefact_external_links_attributes_0_url']").text
assert_equal "", page.find("input[name='artefact[external_links_attributes][0][title]']").value
assert_equal "", page.find("input[name='artefact[external_links_attributes][0][url]']").value

# Link 2 (empty fields)
assert page.has_css?("legend", text: "Link 2")
assert_equal "Title", page.find("label[for='artefact_external_links_attributes_1_title']").text
assert_equal "URL", page.find("label[for='artefact_external_links_attributes_1_url']").text
assert_equal "", page.find("input[name='artefact[external_links_attributes][1][title]']").value
assert_equal "", page.find("input[name='artefact[external_links_attributes][1][url]']").value
end
end

context "Document already has external links when page loads" do
setup do
visit_draft_edition
@draft_edition.artefact.external_links = [{ title: "Link One", url: "https://gov.uk" }]
click_link "Related external links"
end

should "render a pre-populated 'Add another' form" do
# Link 1
assert page.has_css?("legend", text: "Link 1")
assert page.has_css?("input[name='artefact[external_links_attributes][0][_destroy]']")
assert_equal "Title", page.find("label[for='artefact_external_links_attributes_0_title']").text
assert_equal "URL", page.find("label[for='artefact_external_links_attributes_0_url']").text
assert_equal "Link One", page.find("input[name='artefact[external_links_attributes][0][title]']").value
assert_equal "https://gov.uk", page.find("input[name='artefact[external_links_attributes][0][url]']").value

# Link 2 (empty fields)
assert page.has_css?("legend", text: "Link 2")
assert_equal "Title", page.find("label[for='artefact_external_links_attributes_1_title']").text
assert_equal "URL", page.find("label[for='artefact_external_links_attributes_1_url']").text
assert_equal "", page.find("input[name='artefact[external_links_attributes][1][title]']").value
assert_equal "", page.find("input[name='artefact[external_links_attributes][1][url]']").value
end
end

context "User adds a new external link and saves" do
setup do
visit_draft_edition
click_link "Related external links"
end

should "render a prepopulated 'Add another' form" do
within :css, ".gem-c-add-another .js-add-another__fieldset:first-of-type" do
fill_in "Title", with: "A new external link"
fill_in "URL", with: "https://foo.com"
end

click_button("Save")

# Link 1
assert page.has_css?("legend", text: "Link 1")
assert page.has_css?("input[name='artefact[external_links_attributes][0][_destroy]']")
assert_equal "Title", page.find("label[for='artefact_external_links_attributes_0_title']").text
assert_equal "URL", page.find("label[for='artefact_external_links_attributes_0_url']").text
assert_equal "A new external link", page.find("input[name='artefact[external_links_attributes][0][title]']").value
assert_equal "https://foo.com", page.find("input[name='artefact[external_links_attributes][0][url]']").value

# Link 2 (empty fields)
assert page.has_css?("legend", text: "Link 2")
assert_equal "Title", page.find("label[for='artefact_external_links_attributes_1_title']").text
assert_equal "URL", page.find("label[for='artefact_external_links_attributes_1_url']").text
assert_equal "", page.find("input[name='artefact[external_links_attributes][1][title]']").value
assert_equal "", page.find("input[name='artefact[external_links_attributes][1][url]']").value
end
end

context "User deletes an external link and saves" do
setup do
visit_draft_edition
@draft_edition.artefact.external_links = [{ title: "Link One", url: "https://gov.uk" }]
click_link "Related external links"
end

should "render an empty 'Add another' form" do
within :css, ".gem-c-add-another .js-add-another__fieldset:first-of-type" do
check("Delete")
end

click_button("Save")

# Link 1
assert page.has_css?("legend", text: "Link 1")
assert page.has_css?("input[name='artefact[external_links_attributes][0][_destroy]']")
assert_equal "Title", page.find("label[for='artefact_external_links_attributes_0_title']").text
assert_equal "URL", page.find("label[for='artefact_external_links_attributes_0_url']").text
assert_equal "", page.find("input[name='artefact[external_links_attributes][0][title]']").value
assert_equal "", page.find("input[name='artefact[external_links_attributes][0][url]']").value

# Link 2 (empty fields)
assert page.has_css?("legend", text: "Link 2")
assert_equal "Title", page.find("label[for='artefact_external_links_attributes_1_title']").text
assert_equal "URL", page.find("label[for='artefact_external_links_attributes_1_url']").text
assert_equal "", page.find("input[name='artefact[external_links_attributes][1][title]']").value
assert_equal "", page.find("input[name='artefact[external_links_attributes][1][url]']").value
end
end
end

private

def visit_draft_edition
Expand Down
Loading
Loading