-
Notifications
You must be signed in to change notification settings - Fork 0
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
[74] Enter Evaluators A11y Fix #343
base: dev
Are you sure you want to change the base?
Changes from all commits
ddf39d7
bf53f50
77aa9d7
0dcc23b
e4ecf73
ecdf880
d0b4922
c942595
95e863c
0589077
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,46 @@ | ||
import { Controller } from "@hotwired/stimulus" | ||
|
||
// Connects to data-controller="form-validation" | ||
export default class extends Controller { | ||
validatePresence(e) { | ||
const target = e.target; | ||
const formGroup = target.closest(".usa-form-group"); | ||
const fieldName = target.dataset.fieldName || target.id; | ||
const label = this.findLabel(target, formGroup); | ||
|
||
if (!target.value) { | ||
this.addErrorClasses(target, label); | ||
this.updateErrorMessage(fieldName, "can't be blank"); | ||
} else { | ||
this.removeErrorClasses(target, label); | ||
this.updateErrorMessage(fieldName, ""); | ||
} | ||
} | ||
|
||
findLabel(target, formGroup) { | ||
const isSelect = target.tagName === "SELECT" || target.classList.contains("usa-combo-box__input"); | ||
const isRadio = target.type === "radio"; | ||
|
||
const labelId = isSelect ? target.name : target.id; | ||
const labelQuery = isRadio ? "legend" : `label[for="${labelId}"]`; | ||
|
||
return formGroup.querySelector(labelQuery); | ||
} | ||
|
||
addErrorClasses(target, label) { | ||
target.classList.add("border-secondary"); | ||
if (label) label.classList.add("text-secondary"); | ||
} | ||
|
||
removeErrorClasses(target, label) { | ||
target.classList.remove("border-secondary"); | ||
if (label) label.classList.remove("text-secondary"); | ||
} | ||
|
||
updateErrorMessage(field, message) { | ||
const errorElement = document.getElementById(field + "_error"); | ||
if (errorElement) { | ||
errorElement.innerHTML = message; | ||
} | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -8,6 +8,7 @@ def initialize(challenge, phase) | |
end | ||
|
||
def process_evaluator_invitation(email, invitation_params) | ||
@invitation_params = invitation_params | ||
user = User.find_by(email:) | ||
user ? add_existing_user_as_evaluator(user) : handle_invitation(email, invitation_params) | ||
end | ||
|
@@ -44,6 +45,24 @@ def resend_invitation(invitation) | |
|
||
private | ||
|
||
def update_name_for_existing_user(user) | ||
names = @invitation_params[:full_name].to_s.strip.split(/\s+/, 2) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I feel like it's potentially problematic to duplicate this field parsing logic with the EvaluatorInvitation model, in case things change in the future. Can you extract these into a single function? Something like For the validation error message, you can just have a method that adds those errors directly to the model if |
||
if names.length < 2 | ||
return { | ||
success: false, | ||
message: "First and last name are required" | ||
} | ||
end | ||
|
||
first_name, last_name = names | ||
user.update( | ||
first_name:, | ||
last_name: | ||
) | ||
|
||
{ success: true } | ||
end | ||
|
||
def add_existing_user_as_evaluator(user) | ||
if @phase.evaluators.include?(user) | ||
return { | ||
|
@@ -61,6 +80,9 @@ def add_existing_user_as_evaluator(user) | |
} | ||
end | ||
|
||
updated_name = update_name_for_existing_user(user) | ||
return updated_name unless updated_name[:success] | ||
|
||
cpe = ChallengePhasesEvaluator.find_or_create_by(challenge: @challenge, phase: @phase, user:) | ||
|
||
if cpe.persisted? | ||
|
@@ -101,7 +123,8 @@ def create_new_invitation(invitation_params) | |
else | ||
{ | ||
success: false, | ||
message: invitation.errors.full_messages.join(", ") | ||
message: invitation.errors.full_messages.join(", "), | ||
evaluator_invitation: invitation | ||
} | ||
end | ||
end | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm seeing some inconsistency with the error message display depending on the path it takes. Can we try to ensure that if there are any validation issues with the name(s) missing, that the
First & Last Name
label is always highlighted in red (as it does for new users)?existing user
new user
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The first screenshot is also missing the inline error text label describing the issue.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There is also a case where only the first name is entered,
First & Last Name
label should be red: