From 1078aa06ebc0991a3126777161d615c371987ae5 Mon Sep 17 00:00:00 2001 From: PeterHattyar Date: Tue, 20 Aug 2024 14:59:12 +0100 Subject: [PATCH] Breaking up Datepicker to not_before and needed_by inputs MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Date picker arrows in Support app stopped working after the migration to Dart Sass. It’s caused by a known issue in the way Dart Sass handles the icon as Unicode - sass/sass#1395 we were not able to find a workaround. Date picker are not very accessible. When users are filling the dates in the form they normally know them already (and don’t need to pick it by looking at a calendar). Users see a  symbol instead of an arrow. There’s alt text (Next, Prev). It’s also possible to enter date manually. This will allow us to drop dependency on jquery-ui-rails. Due to delivery priorities, we're not converting the entire form into Design System at the moment. --- .../content_advice_requests_controller.rb | 2 +- .../content_change_requests_controller.rb | 2 +- .../remove_user_requests_controller.rb | 2 +- .../support/requests/time_constraint.rb | 11 +- .../_request_details.html.erb | 20 ++- .../_request_details.html.erb | 42 +++++- app/views/support/_time_constraint.html.erb | 34 ++++- spec/features/content_advice_requests_spec.rb | 9 +- spec/features/content_change_requests_spec.rb | 8 +- spec/features/remove_user_requests_spec.rb | 8 +- .../support/requests/time_constraint_spec.rb | 140 ++++++++++++++---- spec/support/app_actions.rb | 8 +- 12 files changed, 234 insertions(+), 52 deletions(-) diff --git a/app/controllers/content_advice_requests_controller.rb b/app/controllers/content_advice_requests_controller.rb index e9b6e5f4a..009203370 100644 --- a/app/controllers/content_advice_requests_controller.rb +++ b/app/controllers/content_advice_requests_controller.rb @@ -20,7 +20,7 @@ def content_advice_request_params :urls, :contact_number, requester_attributes: %i[email name collaborator_emails], - time_constraint_attributes: %i[needed_by_date time_constraint_reason], + time_constraint_attributes: %i[needed_by_date needed_by_day needed_by_month needed_by_year time_constraint_reason], ).to_h end end diff --git a/app/controllers/content_change_requests_controller.rb b/app/controllers/content_change_requests_controller.rb index 6ac5085a0..26e6eaa2d 100644 --- a/app/controllers/content_change_requests_controller.rb +++ b/app/controllers/content_change_requests_controller.rb @@ -22,7 +22,7 @@ def content_change_request_params :url, :related_urls, requester_attributes: %i[email name collaborator_emails], - time_constraint_attributes: %i[not_before_date needed_by_date time_constraint_reason needed_by_time not_before_time], + time_constraint_attributes: %i[not_before_day not_before_month not_before_year needed_by_date needed_by_day needed_by_month needed_by_year time_constraint_reason needed_by_time not_before_time], ).to_h end end diff --git a/app/controllers/remove_user_requests_controller.rb b/app/controllers/remove_user_requests_controller.rb index 56d498f04..c180a4a73 100644 --- a/app/controllers/remove_user_requests_controller.rb +++ b/app/controllers/remove_user_requests_controller.rb @@ -19,7 +19,7 @@ def remove_user_request_params :user_email, :reason_for_removal, requester_attributes: %i[email name collaborator_emails], - time_constraint_attributes: %i[not_before_date needed_by_date time_constraint_reason], + time_constraint_attributes: %i[needed_by_day needed_by_month needed_by_year time_constraint_reason not_before_day not_before_month not_before_year], ).to_h end end diff --git a/app/models/support/requests/time_constraint.rb b/app/models/support/requests/time_constraint.rb index 213b7ca9d..45fde223b 100644 --- a/app/models/support/requests/time_constraint.rb +++ b/app/models/support/requests/time_constraint.rb @@ -4,7 +4,8 @@ module Support module Requests class TimeConstraint include ActiveModel::Model - attr_accessor :not_before_date, :not_before_time, :needed_by_date, :needed_by_time, :time_constraint_reason + include DesignSystemDateHelper + attr_accessor :not_before_day, :not_before_month, :not_before_year, :not_before_time, :needed_by_day, :needed_by_month, :needed_by_year, :needed_by_time, :time_constraint_reason validates_date :needed_by_date, allow_blank: true, on_or_after: :today validates_date :not_before_date, allow_blank: true, on_or_after: :today @@ -24,6 +25,14 @@ class TimeConstraint validates_time :not_before_time, on_or_before: :needed_by_time, unless: proc { |c| [c.needed_by_date, c.needed_by_time, c.not_before_date, c.not_before_time].any?(&:blank?) || c.needed_by_date != c.not_before_date } + + def not_before_date + formatted_date(not_before_day, not_before_month, not_before_year) + end + + def needed_by_date + formatted_date(needed_by_day, needed_by_month, needed_by_year) + end end end end diff --git a/app/views/content_advice_requests/_request_details.html.erb b/app/views/content_advice_requests/_request_details.html.erb index d098ff1e4..3e93575da 100644 --- a/app/views/content_advice_requests/_request_details.html.erb +++ b/app/views/content_advice_requests/_request_details.html.erb @@ -57,11 +57,23 @@
- - <%= r.label :needed_by_date, "Is there a date you need to have a response by?" %> +

+ <%= "Is there a date you need to have a response by?" %> +

+ + + <%= r.label :needed_by_day, "Day", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :needed_by_day, required: false, value: r.object.needed_by_day, class: "govuk-input govuk-date-input__input govuk-input--width-2", id: "needed-by-day" %> - - <%= r.text_field :needed_by_date, placeholder: "dd-mm-yyyy", value: r.object.needed_by_date, class: "input-md-2 form-control", data: { module: "calendar", format: "dd-mm-yy", min_date: 0 } %> + + + <%= r.label :needed_by_month, "Month", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :needed_by_month, required: false, value: r.object.needed_by_month, class: "govuk-input govuk-date-input__input govuk-input--width-2", id: "needed-by-month" %> + + + + <%= r.label :needed_by_year, "Year", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :needed_by_year, required: false, value: r.object.needed_by_year, class: "govuk-input govuk-date-input__input govuk-input--width-4", id: "needed-by-year" %>
diff --git a/app/views/remove_user_requests/_request_details.html.erb b/app/views/remove_user_requests/_request_details.html.erb index 27c29199b..55578afc7 100644 --- a/app/views/remove_user_requests/_request_details.html.erb +++ b/app/views/remove_user_requests/_request_details.html.erb @@ -31,14 +31,44 @@ <%= f.fields_for :time_constraint do |r| %> + +

+ <%= "MUST NOT be removed BEFORE" %> +

+ +
+ For example, 27 3 2007 +
+
- - <%= r.label :not_before_date, "MUST NOT be removed BEFORE" %> - - - <%= r.text_field :not_before_date, required: false, placeholder: "dd-mm-yyyy", value: r.object.not_before_date, class: "input-md-2 form-control", data: { module: "calendar", format: "dd-mm-yy", min_date: 0 } %> - + +
+ +
+
+ <%= r.label :not_before_day, "Day", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :not_before_day, required: false, value: r.object.not_before_day, class: "govuk-input govuk-date-input__input govuk-input--width-2", id: "not-before-day" %> +
+
+ +
+
+ <%= r.label :not_before_month, "Month", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :not_before_month, required: false, value: r.object.not_before_month, class: "govuk-input govuk-date-input__input govuk-input--width-2", id: "not-before-month" %> +
+
+ +
+
+ <%= r.label :not_before_year, "Year", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :not_before_year, required: false, value: r.object.not_before_year, class: "govuk-input govuk-date-input__input govuk-input--width-4", id: "not-before-year" %> +
+
+ +
+
+ <% end %>
diff --git a/app/views/support/_time_constraint.html.erb b/app/views/support/_time_constraint.html.erb index 9db761f7f..7c5981b55 100644 --- a/app/views/support/_time_constraint.html.erb +++ b/app/views/support/_time_constraint.html.erb @@ -8,9 +8,22 @@ <%= r.label :needed_by_date, "Deadline" %> - - <%= r.text_field :needed_by_date, required: false, placeholder: "dd-mm-yyyy", value: r.object.needed_by_date, class: "input-md-2 form-control", data: { module: "calendar", format: "dd-mm-yy", min_date: 0 } %> +
+ + + <%= r.label :needed_by_day, "Day", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :needed_by_day, required: false, value: r.object.needed_by_day, class: "govuk-input govuk-date-input__input govuk-input--width-2", id: "needed-by-day" %> + + + + <%= r.label :needed_by_month, "Month", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :needed_by_month, required: false, value: r.object.needed_by_month, class: "govuk-input govuk-date-input__input govuk-input--width-2", id: "needed-by-month" %> + + + <%= r.label :needed_by_year, "Year", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :needed_by_year, required: false, value: r.object.needed_by_year, class: "govuk-input govuk-date-input__input govuk-input--width-4", id: "needed-by-year" %> +
@@ -26,8 +39,21 @@ <%= r.label :not_before_date, "Must not be published before" %> - - <%= r.text_field :not_before_date, required: false, placeholder: "dd-mm-yyyy", value: r.object.not_before_date, class: "input-md-2 form-control", data: { module: "calendar", format: "dd-mm-yy", min_date: 0 } %> +
+ + + <%= r.label :not_before_day, "Day", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :not_before_day, required: false, value: r.object.not_before_day, class: "govuk-input govuk-date-input__input govuk-input--width-2", id: "not-before-day" %> + + + + <%= r.label :not_before_month, "Month", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :not_before_month, required: false, value: r.object.not_before_month, class: "govuk-input govuk-date-input__input govuk-input--width-2", id: "not-before-month" %> + + + + <%= r.label :not_before_year, "Year", class: "govuk-label govuk-date-input__label" %> + <%= r.text_field :not_before_year, required: false, value: r.object.not_before_year, class: "govuk-input govuk-date-input__input govuk-input--width-4", id: "not-before-year" %>
diff --git a/spec/features/content_advice_requests_spec.rb b/spec/features/content_advice_requests_spec.rb index b97a30176..394ef69eb 100644 --- a/spec/features/content_advice_requests_spec.rb +++ b/spec/features/content_advice_requests_spec.rb @@ -37,7 +37,9 @@ title: "Which format", details: "I need help to choose a format, here's my content...", urls: "https://www.gov.uk/x, https://www.gov.uk/y", - needed_by: "12-01-#{next_year}", + needed_by_day: "12", + needed_by_month: "01", + needed_by_year: next_year.to_s, reason_for_deadline: "Ministerial announcement Z", contact_number: "0121 111111", ) @@ -81,8 +83,9 @@ def user_requests_content_advice(details) fill_in "Title of request", with: details[:title] fill_in "Please explain what you would like help with", with: details[:details] fill_in "Relevant URLs (if applicable)", with: details[:urls] - - fill_in "Is there a date you need to have a response by?", with: details[:needed_by] + find("#needed-by-day").set(details[:needed_by_day]) + find("#needed-by-month").set(details[:needed_by_month]) + find("#needed-by-year").set(details[:needed_by_year]) fill_in "Reason for deadline", with: details[:reason_for_deadline] fill_in "Contact telephone number (in case we need to call you to discuss the content)", with: details[:contact_number] diff --git a/spec/features/content_change_requests_spec.rb b/spec/features/content_change_requests_spec.rb index 784728373..a047c15a8 100644 --- a/spec/features/content_change_requests_spec.rb +++ b/spec/features/content_change_requests_spec.rb @@ -61,9 +61,13 @@ details_of_change: "Out of date XX YY", url: "http://gov.uk/X", related_urls: "http://gov.uk/welsh", - needed_by_date: "31-12-#{next_year}", + needed_by_day: "31", + needed_by_month: "12", + needed_by_year: next_year.to_s, needed_by_time: "13:00", - not_before_date: "01-12-#{next_year}", + not_before_day: "01", + not_before_month: "12", + not_before_year: next_year.to_s, not_before_time: "18:00", reason: "New law", ) diff --git a/spec/features/remove_user_requests_spec.rb b/spec/features/remove_user_requests_spec.rb index f8d435714..d4737a683 100644 --- a/spec/features/remove_user_requests_spec.rb +++ b/spec/features/remove_user_requests_spec.rb @@ -35,7 +35,9 @@ user_name: "Bob", user_email: "bob@someagency.gov.uk", reason_for_removal: "User has left the organisation", - not_before_date: "31-12-#{next_year}", + not_before_day: "31", + not_before_month: "12", + not_before_year: next_year.to_s, ) expect(request).to have_been_made @@ -56,7 +58,9 @@ def user_requests_removal_of_another_user(details) fill_in "Reason for removal", with: details[:reason_for_removal] end - fill_in "MUST NOT be removed BEFORE", with: details[:not_before_date] + find("#not-before-day").set(details[:not_before_day]) + find("#not-before-month").set(details[:not_before_month]) + find("#not-before-year").set(details[:not_before_year]) user_submits_the_request_successfully end diff --git a/spec/models/support/requests/time_constraint_spec.rb b/spec/models/support/requests/time_constraint_spec.rb index 7fb3de8c5..59808798d 100644 --- a/spec/models/support/requests/time_constraint_spec.rb +++ b/spec/models/support/requests/time_constraint_spec.rb @@ -11,26 +11,88 @@ def time_as_str(time) time.strftime("%H:%M") end - it { should_not allow_value("xxx").for(:needed_by_date) } + it "should not allow needed by date values to be random characters " do + constraint = TimeConstraint.new( + needed_by_day: "xxx", + needed_by_month: "xxx", + needed_by_year: "xxx", + ) + expect(constraint).to_not be_valid + end - it { should allow_value(as_str(Date.tomorrow)).for(:needed_by_date) } - it { should allow_value(as_str(Time.zone.today)).for(:needed_by_date) } - it { should_not allow_value(as_str(Date.yesterday)).for(:needed_by_date) } + it "allows the 'needed by' date to be in the future" do + constraint = TimeConstraint.new( + needed_by_day: Time.zone.tomorrow.strftime("%d"), + needed_by_month: Time.zone.tomorrow.strftime("%m"), + needed_by_year: Time.zone.tomorrow.strftime("%Y"), + ) + expect(constraint).to be_valid + end - it { should_not allow_value("xxx").for(:not_before_date) } + it "allows the 'needed by' date to be today" do + constraint = TimeConstraint.new( + needed_by_day: Time.zone.today.strftime("%d"), + needed_by_month: Time.zone.today.strftime("%m"), + needed_by_year: Time.zone.today.strftime("%Y"), + ) + expect(constraint).to be_valid + end - it { should allow_value(as_str(Date.tomorrow)).for(:not_before_date) } - it { should allow_value(as_str(Time.zone.today)).for(:not_before_date) } - it { should_not allow_value(as_str(Date.yesterday)).for(:not_before_date) } + it "should not allow the 'needed by' date to be in the past" do + constraint = TimeConstraint.new( + needed_by_day: Time.zone.yesterday.strftime("%d"), + needed_by_month: Time.zone.yesterday.strftime("%m"), + needed_by_year: Time.zone.yesterday.strftime("%Y"), + ) + expect(constraint).to_not be_valid + end + + it "allows the 'not before' date to be in the future" do + constraint = TimeConstraint.new( + not_before_day: Time.zone.tomorrow.strftime("%d"), + not_before_month: Time.zone.tomorrow.strftime("%m"), + not_before_year: Time.zone.tomorrow.strftime("%Y"), + ) + expect(constraint).to be_valid + end + + it "allows the 'not before' date to be today" do + constraint = TimeConstraint.new( + not_before_day: Time.zone.today.strftime("%d"), + not_before_month: Time.zone.today.strftime("%m"), + not_before_year: Time.zone.today.strftime("%Y"), + ) + expect(constraint).to be_valid + end + + it "doesn't allow the 'not before' date to be in the past" do + constraint = TimeConstraint.new( + not_before_day: Time.zone.yesterday.strftime("%d"), + not_before_month: Time.zone.yesterday.strftime("%m"), + not_before_year: Time.zone.yesterday.strftime("%Y"), + ) + expect(constraint).to_not be_valid + end - it "allows the 'not before' and 'needed by' dates to be blank" do - expect(TimeConstraint.new(not_before_date: "", needed_by_date: "")).to be_valid + it "allows the 'not before' and 'needed by' date fields to be blank" do + expect(TimeConstraint.new( + not_before_day: "", + not_before_month: "", + not_before_year: "", + needed_by_day: "", + needed_by_month: "", + needed_by_year: "", + )).to be_valid end it "doesn't allow the 'not before' date to be set after the 'needed by' date" do constraint = TimeConstraint.new( - not_before_date: as_str(Date.tomorrow + 1.day), - needed_by_date: as_str(Date.tomorrow), + not_before_day: (Time.zone.tomorrow + 1.day).strftime("%d"), + not_before_month: (Time.zone.tomorrow + 1.day).strftime("%m"), + not_before_year: (Time.zone.tomorrow + 1.day).strftime("%Y"), + needed_by_day: Time.zone.tomorrow.strftime("%d"), + needed_by_month: Time.zone.tomorrow.strftime("%m"), + needed_by_year: Time.zone.tomorrow.strftime("%Y"), ) expect(constraint).to_not be_valid expect(constraint).to have_at_least(1).error_on(:not_before_date) @@ -38,8 +100,12 @@ def time_as_str(time) it "doesn't allow the 'not before' time to be set after the 'needed by' time, if the 'not before' date and 'not before' time are the same" do constraint = TimeConstraint.new( - not_before_date: as_str(Date.tomorrow), - needed_by_date: as_str(Date.tomorrow), + not_before_day: Time.zone.tomorrow.strftime("%d"), + not_before_month: Time.zone.tomorrow.strftime("%m"), + not_before_year: Time.zone.tomorrow.strftime("%Y"), + needed_by_day: Time.zone.tomorrow.strftime("%d"), + needed_by_month: Time.zone.tomorrow.strftime("%m"), + needed_by_year: Time.zone.tomorrow.strftime("%Y"), not_before_time: "12:00", needed_by_time: "11:00", ) @@ -49,8 +115,12 @@ def time_as_str(time) it "allows the 'not before' time to be set after the 'needed by' time, if the 'not before' date and 'not before' time are different" do constraint = TimeConstraint.new( - not_before_date: as_str(Date.tomorrow), - needed_by_date: as_str(Date.tomorrow + 1.day), + not_before_day: Time.zone.tomorrow.strftime("%d"), + not_before_month: Time.zone.tomorrow.strftime("%m"), + not_before_year: Time.zone.tomorrow.strftime("%Y"), + needed_by_day: (Time.zone.tomorrow + 1.day).strftime("%d"), + needed_by_month: (Time.zone.tomorrow + 1.day).strftime("%m"), + needed_by_year: (Time.zone.tomorrow + 1.day).strftime("%Y"), not_before_time: "12:00", needed_by_time: "11:00", ) @@ -59,14 +129,18 @@ def time_as_str(time) it "validates that the 'not before' time is after now" do constraint = TimeConstraint.new( - not_before_date: as_str(Time.zone.today), + not_before_day: Time.zone.today.strftime("%d"), + not_before_month: Time.zone.today.strftime("%m"), + not_before_year: Time.zone.today.strftime("%Y"), not_before_time: time_as_str(Time.zone.now - 1.minute), ) expect(constraint).to_not be_valid expect(constraint).to have_at_least(1).error_on(:not_before_time) constraint = TimeConstraint.new( - not_before_date: as_str(Date.tomorrow), + not_before_day: Time.zone.tomorrow.strftime("%d"), + not_before_month: Time.zone.tomorrow.strftime("%m"), + not_before_year: Time.zone.tomorrow.strftime("%Y"), not_before_time: time_as_str(Time.zone.now - 1.minute), ) expect(constraint).to be_valid @@ -74,27 +148,39 @@ def time_as_str(time) it "validates that the 'needed by' time is after now" do constraint = TimeConstraint.new( - needed_by_date: as_str(Time.zone.today), + needed_by_day: Time.zone.today.strftime("%d"), + needed_by_month: Time.zone.today.strftime("%m"), + needed_by_year: Time.zone.today.strftime("%Y"), needed_by_time: time_as_str(Time.zone.now - 1.minute), ) expect(constraint).to_not be_valid expect(constraint).to have_at_least(1).error_on(:needed_by_time) constraint = TimeConstraint.new( - needed_by_date: as_str(Time.zone.tomorrow), + needed_by_day: Time.zone.tomorrow.strftime("%d"), + needed_by_month: Time.zone.tomorrow.strftime("%m"), + needed_by_year: Time.zone.tomorrow.strftime("%Y"), needed_by_time: time_as_str(Time.zone.now - 1.minute), ) expect(constraint).to be_valid end it "allows a blank not_before_date if the needed_by_date is set" do - expect(TimeConstraint.new(needed_by_date: as_str(Date.tomorrow))).to be_valid + expect(TimeConstraint.new( + needed_by_day: Time.zone.tomorrow.strftime("%d"), + needed_by_month: Time.zone.tomorrow.strftime("%m"), + needed_by_year: Time.zone.tomorrow.strftime("%Y"), + )).to be_valid end it "allows launch dates (i.e. not_before_date = needed_by_date)" do constraint = TimeConstraint.new( - not_before_date: as_str(Date.tomorrow), - needed_by_date: as_str(Date.tomorrow), + not_before_day: Time.zone.tomorrow.strftime("%d"), + not_before_month: Time.zone.tomorrow.strftime("%m"), + not_before_year: Time.zone.tomorrow.strftime("%Y"), + needed_by_day: Time.zone.tomorrow.strftime("%d"), + needed_by_month: Time.zone.tomorrow.strftime("%m"), + needed_by_year: Time.zone.tomorrow.strftime("%Y"), ) expect(constraint).to be_valid end @@ -117,8 +203,12 @@ def time_as_str(time) it "returns the correct error message when 'not before' date is after 'needed by' date" do constraint = Support::Requests::TimeConstraint.new( - not_before_date: (Time.zone.today + 2.days).strftime("%d-%m-%Y"), - needed_by_date: (Time.zone.today + 1.day).strftime("%d-%m-%Y"), + not_before_day: (Time.zone.today + 2.days).strftime("%d"), + not_before_month: (Time.zone.today + 2.days).strftime("%m"), + not_before_year: (Time.zone.today + 2.days).strftime("%Y"), + needed_by_day: (Time.zone.today + 1.day).strftime("%d"), + needed_by_month: (Time.zone.today + 1.day).strftime("%m"), + needed_by_year: (Time.zone.today + 1.day).strftime("%Y"), ) constraint.valid? diff --git a/spec/support/app_actions.rb b/spec/support/app_actions.rb index e136a58ba..2832c0f36 100644 --- a/spec/support/app_actions.rb +++ b/spec/support/app_actions.rb @@ -83,9 +83,13 @@ def doctype_summary_results end def user_fills_out_time_constraints(details) - fill_in "Deadline", with: details[:needed_by_date] + find("#needed-by-day").set(details[:needed_by_day]) + find("#needed-by-month").set(details[:needed_by_month]) + find("#needed-by-year").set(details[:needed_by_year]) fill_in "Time this must be published by", with: details[:needed_by_time] - fill_in "Must not be published before", with: details[:not_before_date] + find("#not-before-day").set(details[:not_before_day]) + find("#not-before-month").set(details[:not_before_month]) + find("#not-before-year").set(details[:not_before_year]) fill_in "Time this must not be published before", with: details[:not_before_time] fill_in "Reason for deadline", with: details[:reason] end