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

[179] View Evaluator's Submissions #258

Merged
merged 61 commits into from
Dec 9, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
61 commits
Select commit Hold shift + click to select a range
7abaa15
245 | Update invited evaluator user setup
emmabjj Nov 4, 2024
2093119
245 | Move evaluator invitation to evaluator user logic
emmabjj Nov 5, 2024
6771cf4
245 | Add check for evaluator's role
emmabjj Nov 5, 2024
ef02ade
245 | Remove duplicate logic and handle in/valid evaluator roles
emmabjj Nov 5, 2024
23e5701
245 | Add successfully removed flash notice back in
emmabjj Nov 5, 2024
27f35d0
245 | Update tests
emmabjj Nov 6, 2024
e64fa23
245 | CPE created automatically, adjust redudant conditional
emmabjj Nov 6, 2024
1d36e0d
179 | UI/Add view for evaluator submissions
emmabjj Nov 7, 2024
d576fa0
179 | Add UI for the unassign modal and js
emmabjj Nov 7, 2024
a91e88a
179 | Add routes
emmabjj Nov 7, 2024
a4a9298
179 | Add evaluator_submissions_assignments association to phase thro…
emmabjj Nov 7, 2024
742af61
179 | Add evaluation submissions controller
emmabjj Nov 7, 2024
44c2db5
179 | Add evaluation status text colors
emmabjj Nov 7, 2024
145886a
179 | Add evaluation status order
emmabjj Nov 7, 2024
16866fb
Merge branch '200_manage_evaluators_list' into 179_evaluator_submissi…
emmabjj Nov 7, 2024
4295f4d
179 | Add evaluator submissions message translations
emmabjj Nov 7, 2024
3f803d3
Merge branch '200_manage_evaluators_list' into 179_evaluator_submissi…
emmabjj Nov 19, 2024
8775c82
179 | Update routes
emmabjj Nov 22, 2024
a68e0cb
179 | Update view and use partials
emmabjj Nov 22, 2024
5c5ad7e
179 | Use update for reassign & unassign
emmabjj Nov 22, 2024
e86b23e
179 | Add recused_unassigned status
emmabjj Nov 22, 2024
fbe433a
179 | Back link partial
emmabjj Nov 22, 2024
3ff0089
179 | Adjust js to use patch update on status reassign/unassign
emmabjj Nov 22, 2024
e7882bf
179 | Add to translations
emmabjj Nov 22, 2024
9c89936
Small updates
emmabjj Nov 22, 2024
01b132f
Small fix
emmabjj Nov 22, 2024
6c32c1d
Merge branch 'dev' into 179_evaluator_submissions_view
emmabjj Nov 22, 2024
1a6531e
Update order by status on evaluation submission assignments
emmabjj Nov 22, 2024
3da992d
Remove unused argument
emmabjj Nov 22, 2024
e541c13
Update ordered by status
emmabjj Nov 22, 2024
ae2870f
77 | Add recused alert for managers on evaluator submission assignmen…
emmabjj Nov 22, 2024
25f0d6d
Merge branch 'dev' into 179_evaluator_submissions_view
emmabjj Nov 22, 2024
8e207e1
179 | Display score for evaluator submission assignments
emmabjj Nov 22, 2024
f0013f6
179 | Adjust SQL statement for status order
emmabjj Nov 22, 2024
6602fb3
179 | Scope score to evaluator's evaluation score
emmabjj Nov 22, 2024
9f44ad8
179 | Recused unassigned cannot be reassign to the submission
emmabjj Nov 22, 2024
2416337
77 | Update recused conditional
emmabjj Nov 25, 2024
d029c20
179 | Update route for evaluation submission assignments
emmabjj Nov 25, 2024
c890db4
179 | Update js to use assignment
emmabjj Nov 25, 2024
01209a4
179 | Update the display score to check for assignment completion
emmabjj Nov 25, 2024
ec365f9
179 | Update ordered by status query
emmabjj Nov 25, 2024
575eda4
179 | Update tests wip
emmabjj Nov 25, 2024
f824247
179 | Remove unused argument in display_score
emmabjj Nov 25, 2024
d99a8b4
179 | Update tests for display scores
emmabjj Nov 25, 2024
28e77ea
Merge branch 'dev' into 179_evaluator_submissions_view
emmabjj Nov 25, 2024
2083129
179 | Update flash, closing date, and error status
emmabjj Dec 3, 2024
f0b582b
179 | Add tests, update statuses, and colors
emmabjj Dec 5, 2024
ffd144d
Rename stat summary for evaluation submission assignments
emmabjj Dec 6, 2024
0edad56
Merge branch '179_evaluator_submissions_view' into 77_recused_submiss…
emmabjj Dec 6, 2024
9123d76
179 | Adjust sorting scope and evaluation status
emmabjj Dec 9, 2024
9d4f6d9
update status colors
stepchud Dec 9, 2024
447a75f
quick syntax fix
emmabjj Dec 9, 2024
3f523f5
Merge remote-tracking branch 'origin/179_evaluator_submissions_view' …
stepchud Dec 9, 2024
3f12ec5
Merge remote-tracking branch 'origin/179_evaluator_submissions_view' …
stepchud Dec 9, 2024
9cac42d
179 | Adjust sorting scope and evaluation status
emmabjj Dec 9, 2024
2224b72
quick syntax fix
emmabjj Dec 9, 2024
b48c371
Merge branch '179_evaluator_submissions_view' into 77_recused_submiss…
emmabjj Dec 9, 2024
a77ad0f
Disable rubocop check on flash before render
emmabjj Dec 9, 2024
f8c5f77
Update .codeclimate.yml rubocop name
stepchud Dec 9, 2024
018dc76
Update app/views/evaluator_submission_assignments/_unassigned_submiss…
stepchud Dec 9, 2024
796252d
Merge pull request #286 from GSA/77_recused_submission_alert
emmabjj Dec 9, 2024
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
3 changes: 3 additions & 0 deletions .codeclimate.yml
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,9 @@ plugins:
rubocop:
enabled: true
channel: rubocop-1-56-3
checks:
Rubocop/Rails/ActionControllerFlashBeforeRender:
enabled: false
exclude_patterns:
- .nix-bundler
- config/
Expand Down
115 changes: 115 additions & 0 deletions app/controllers/evaluator_submission_assignments_controller.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,115 @@
# frozen_string_literal: true

# Controller for evaluator submissions assignments index and update status
class EvaluatorSubmissionAssignmentsController < ApplicationController
stepchud marked this conversation as resolved.
Show resolved Hide resolved
before_action -> { authorize_user('challenge_manager') }
before_action :set_challenge_and_phase
before_action :set_evaluator, only: [:index]
before_action :set_assignment, only: [:update]

def index
@evaluator_assignments = @phase.evaluator_submission_assignments.includes(:submission).where(user_id: @evaluator.id)
@assigned_submissions = @evaluator_assignments.
where(status: %i[assigned recused]).
includes(:evaluation).
ordered_by_status
@unassigned_submissions = @evaluator_assignments.
where(status: %i[unassigned recused_unassigned]).
ordered_by_status
@submissions_count = calculate_submissions_count(@assigned_submissions)
end

# update only the status of the evaluation submission assignment to unassign or reassign an evaluator
def update
new_status = status_from_params

unless valid_status?(new_status)
return render_invalid_status_error
end

if update_assignment_status(new_status)
handle_successful_update(new_status)
else
handle_failed_update(new_status)
end
end

private

def set_challenge_and_phase
@phase = Phase.where(challenge: current_user.challenge_manager_challenges).find(params[:phase_id])
@challenge = @phase.challenge
end

def set_evaluator
@evaluator = @phase.evaluators.find(params[:evaluator_id])
end

def set_assignment
@assignment = @phase.evaluator_submission_assignments.find(params[:id])
end

def status_from_params
status = params[:status] || params.dig(:evaluator_submission_assignment, :status)
status&.to_sym
end

def valid_status?(status)
EvaluatorSubmissionAssignment.statuses.keys.map(&:to_sym).include?(status)
end

def render_invalid_status_error
render json: { success: false, message: 'Invalid status' }, status: :unprocessable_entity
end

def update_assignment_status(new_status)
@assignment.update(status: new_status)
end

def handle_successful_update(new_status)
flash[:success] = t("evaluator_submission_assignments.#{new_status}.success")
respond_to do |format|
format.html { redirect_to_assignment_path }
format.json { render json: { success: true, message: flash[:success] } }
end
end

def handle_failed_update(new_status)
flash[:error] = t("evaluator_submission_assignments.#{new_status}.failure")
respond_to do |format|
format.html { redirect_to_assignment_path }
format.json { render json: { success: false, message: flash[:error] }, status: :unprocessable_entity }
end
end

def redirect_to_assignment_path
redirect_to phase_evaluator_submission_assignments_path(
@phase,
evaluator_id: params[:evaluator_id]
)
end

def calculate_submissions_count(assignments)
counts = count_by_status(assignments)
counts.merge("total" => calculate_total(counts))
end

def count_by_status(assignments)
{
"completed" => count_completed(assignments),
"in_progress" => count_in_progress(assignments),
"not_started" => count_not_started(assignments),
"recused" => count_recused(assignments)
}
end

def count_completed(assignments) = assignments.count { |a| a.evaluation&.completed_at.present? }

def count_in_progress(assignments) = assignments.count { |a| a.evaluation.present? && a.evaluation.completed_at.nil? }

def count_not_started(assignments) = assignments.count { |a| a.assigned? && a.evaluation.nil? }

def count_recused(assignments) = assignments.count(&:recused?)

def calculate_total(counts) = counts.values.sum
end
48 changes: 35 additions & 13 deletions app/helpers/evaluators_helper.rb
Original file line number Diff line number Diff line change
Expand Up @@ -2,22 +2,44 @@

# View helpers for rendering users with the evaluator role.
module EvaluatorsHelper
STATUS_COLORS = {
not_started: 'bg-error-dark',
in_progress: 'bg-accent-warm-dark',
completed: 'bg-success-dark',
recused: 'bg-base',
unassigned: 'bg-base',
recused_unassigned: 'bg-base'
}.freeze

def user_status(evaluator)
if evaluator.is_a?(User)
evaluator.status == 'active' ? "Available" : "Awaiting Approval"
else
"Invite Sent"
end
return "Invite Sent" unless evaluator.is_a?(User)

evaluator.status == 'active' ? "Available" : "Awaiting Approval"
end

def assigned_submissions_count(evaluator, challenge, phase)
if evaluator.is_a?(User)
evaluator.evaluator_submission_assignments.
joins(:submission).
where(submissions: { challenge:, phase: }).
count
else
0
end
return 0 unless evaluator.is_a?(User)

evaluator.evaluator_submission_assignments.
joins(:submission).
where(submissions: { challenge:, phase: }).
where(status: :assigned).
count
end

def evaluation_submission_assignment_color(assignment)
status = if assignment.is_a?(EvaluatorSubmissionAssignment)
assignment.evaluation_status
else
assignment.to_sym
end

STATUS_COLORS[status]
end

def display_score(assignment)
return 'N/A' unless assignment.evaluation_status == :completed

assignment.evaluation&.total_score || 'N/A'
end
end
3 changes: 3 additions & 0 deletions app/javascript/controllers/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,3 +11,6 @@ application.register("evaluation-criteria", EvaluationCriteriaController);

import DeleteEvaluatorModalController from "./delete_evaluator_modal_controller"
application.register("delete-evaluator-modal", DeleteEvaluatorModalController)

import UnassignEvaluatorSubmissionModalController from "./unassign_evaluator_submission_modal_controller"
application.register("unassign-evaluator-submission-modal", UnassignEvaluatorSubmissionModalController)
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
import { Controller } from "@hotwired/stimulus"

export default class extends Controller {
static targets = ["modal"]
static values = {
phaseId: String,
assignmentId: String
}

connect() {
this.modalTarget.addEventListener('click', this.handleOutsideClick.bind(this));
}

disconnect() {
this.modalTarget.removeEventListener('click', this.handleOutsideClick.bind(this));
}

open(event) {
event.preventDefault();
this.setValues(event.currentTarget.dataset);
this.modalTarget.showModal();
}

close() {
this.modalTarget.close();
}

handleOutsideClick(event) {
if (event.target === this.modalTarget) {
this.close();
}
}

setValues(dataset) {
this.assignmentIdValue = dataset.assignmentId;
this.phaseIdValue = dataset.phaseId;
}

unassignEvaluatorSubmission() {
const csrfToken = document.querySelector('meta[name="csrf-token"]').content;
fetch(`/phases/${this.phaseIdValue}/evaluator_submission_assignments/${this.assignmentIdValue}`, {
method: 'PATCH',
headers: {
'Content-Type': 'application/json',
'X-CSRF-Token': csrfToken,
'Accept': 'application/json'
},
body: JSON.stringify({
evaluator_submission_assignment: { status: 'unassigned' }
})
})
.then(response => response.json())
.then(data => {
if (data.success) {
const evaluatorId = new URLSearchParams(window.location.search).get('evaluator_id');
window.location.href = `/phases/${this.phaseIdValue}/evaluator_submission_assignments?evaluator_id=${evaluatorId}`;
Comment on lines +55 to +56
Copy link
Contributor

@stepchud stepchud Nov 27, 2024

Choose a reason for hiding this comment

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

this is essentially doing location.reload() right? I have a different idea, what if you redirect to the path provided in the response data. you could render the url in the response JSON as data.redirect_to or something and then reuse that same redirect path for both success and error since the backend will determine where to send the user. I think this will also display the flash message that was stored after the browser redirects, so you may not need the alert() either. if you want you can still log a console.error() for developer debugging help messages.

Suggested change
const evaluatorId = new URLSearchParams(window.location.search).get('evaluator_id');
window.location.href = `/phases/${this.phaseIdValue}/evaluator_submission_assignments?evaluator_id=${evaluatorId}`;
window.location.href = data.redirect_to;

} else {
throw new Error(data.message || 'Failed to unassign evaluator from submission');
}
})
.catch(error => {
console.error('Error:', error);
alert(error.message || 'An error occurred while unassigning the evaluator from the submission');
});
}
}
44 changes: 43 additions & 1 deletion app/models/evaluator_submission_assignment.rb
Original file line number Diff line number Diff line change
Expand Up @@ -12,9 +12,51 @@
# updated_at :datetime not null
#
class EvaluatorSubmissionAssignment < ApplicationRecord
ORDER_VALUES = {
recused: 0,
unassigned: 1,
recused_unassigned: 2,
not_started: 3,
in_progress: 4,
completed: 5
}.freeze

# Associations
belongs_to :submission
belongs_to :evaluator, class_name: "User", foreign_key: :user_id, inverse_of: :assigned_submissions
has_one :evaluation, dependent: :destroy

enum :status, { assigned: 0, unassigned: 1, recused: 2 }
has_one :phase, through: :submission

enum :status, {
assigned: 0,
unassigned: 1,
recused: 2,
recused_unassigned: 3
}

def self.ordered_by_status
select('evaluator_submission_assignments.*, evaluations.id AS evaluation_id, evaluations.completed_at').
left_joins(:evaluation).
to_a.
sort_by { |assignment| ORDER_VALUES[assignment.evaluation_status] }
end

def evaluation_status
return status.to_sym unless assigned?

assigned_evaluation_status
end

private

def assigned_evaluation_status
if evaluation&.completed_at.present?
:completed
elsif evaluation.present?
:in_progress
else
:not_started
end
end
end
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
<div
class="usa-summary-box bg-base-lightest border-1px border-base-dark radius-0 margin-bottom-2"
role="region"
aria-labelledby="summary-box-key-information"
>
<div class="usa-summary-box__body">
<div class="display-flex flex-column tablet:flex-row tablet:flex-align-center">
<div class="tablet:flex-fill">
<div class="display-flex flex-align-center margin-bottom-2 tablet:margin-bottom-0">
<div class="margin-right-2">
<span class="font-sans-3xl text-primary text-bold"><%= @assigned_submissions.count %></span>
</div>
<div>
<h3 class="usa-summary-box__heading text-primary margin-bottom-05" id="summary-box-key-information">
Assigned Submissions
</h3>
<p class="usa-summary-box__text text-primary-darker text-bold">
Evaluations due by <%= @phase.evaluation_form.closing_date.strftime('%m/%d/%Y') %>
</p>
</div>
</div>
</div>

<div class="border-top border-base-dark margin-y-2 tablet:display-none"></div>

<div class="display-flex flex-row tablet:margin-left-auto">
<div class="text-center margin-right-5">
<span class="font-sans-xl text-success-dark text-bold"><%= @submissions_count["completed"] || 0 %></span><br>
<span class="font-sans-lg text-success-dark text-bold">Completed</span>
</div>
<div class="text-center margin-right-4">
<span class="font-sans-xl text-accent-warm-dark text-bold"><%= @submissions_count["in_progress"] || 0 %></span><br>
<span class="font-sans-lg text-accent-warm-dark text-bold">In Progress</span>
</div>
<div class="text-center margin-right-2">
<span class="font-sans-xl text-error-dark text-bold"><%= @submissions_count["not_started"] || 0 %></span><br>
<span class="font-sans-lg text-error-dark text-bold">Not Started</span>
</div>
</div>
</div>
</div>
</div>
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
<tr data-evaluator-id="<%= evaluator.id %>" data-evaluator-type="user">
<td data-label="Submission ID" class="text-top text-bold">
<% if assignment.status == "recused" %>
<%= image_tag('images/usa-icons/error.svg', class: "usa-icon--size-3", alt: "Recused", style: "vertical-align: middle; margin-right: 5px;") %>
<% end %>
<%= assignment.submission.id %>
</td>
<td data-label="Evaluation status">
<span class="text-top usa-tag <%= evaluation_submission_assignment_color(assignment) %>"><%= assignment.evaluation_status.to_s.titleize %></span>
</td>
<td data-label="Score" class="text-top text-normal"><%= display_score(assignment) %></td>
<td data-label="">
<div class="display-flex flex-justify-end">
<% if assignment.evaluation_status == :completed %>
<%= link_to "View Evaluation", submissions_path(@challenge), class: 'usa-button font-body-3xs margin-right-1' %>
<% end %>
<%= button_tag "Unassign", type: 'button', class: 'usa-button usa-button--outline font-body-3xs', data: {
action: "click->unassign-evaluator-submission-modal#open",
assignment_id: assignment.id,
phase_id: @phase.id
} %>
</div>
</td>
</tr>
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
<div class="usa-table-container--scrollable width-full margin-bottom-10" tabindex="0">
<table class="usa-table usa-table--stacked-header usa-table--borderless width-full gray-header">
<thead>
<tr>
<th scope="col">Submission ID</th>
<th scope="col" class="text-black">Evaluation status</th>
<th scope="col">Score</th>
<th scope="col" aria-label="Actions"></th>
</tr>
</thead>
<tbody>
<%= yield %>
</tbody>
</table>
</div>
Loading
Loading