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

Gsa/production release #731

Merged
merged 4 commits into from
Jan 14, 2025
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
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 training-front-end/src/components/AdminSearchUser.vue
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,9 @@
function cancelEdit(){
setSelectedUser(undefined)
setSelectedAgencyId(undefined)
// refresh search on cancel from edit reporting page as you can also update user details on the page without
// updating the reporting access. Refresh page to display updated details.
search()
}

async function updateUserSuccess(message) {
Expand Down
3 changes: 2 additions & 1 deletion training-front-end/src/components/AdminTrainingReport.vue
Original file line number Diff line number Diff line change
Expand Up @@ -58,7 +58,7 @@

//format dates from uswds standard to the format needed for backend
const formatDateToYYYYMMDD = (dates) => {
return dates ? dates.map(date => (date ? new Date(date).toISOString().split('T')[0] : null)) : [];
return dates ? dates.map(date => (date ? new Date(date).toISOString() : null)) : [];
};

async function downloadReport() {
Expand Down Expand Up @@ -133,6 +133,7 @@
:validator="v_all_info$.quiz_names"
name="Quiz type(s)"
legend="Quiz type(s)"
class="margin-top-4"
/>
<USWDSComboBox
v-model="user_input.agency_id"
Expand Down
29 changes: 29 additions & 0 deletions training-front-end/src/components/ReportRepository.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
<script>

import {useStore} from "@nanostores/vue";
import {profile} from "../stores/user.js";

const user = useStore(profile)
const base_url = import.meta.env.PUBLIC_API_BASE_URL
const downloadTrainingCompletionReport = async function(filterData){
const response = await fetch(`${base_url}/api/v1/users/download-smartpay-training-report`, {
method: 'POST',
headers: {
'Authorization': `Bearer ${user.value.jwt}`,
'Content-Type': 'application/json'
},
body: JSON.stringify(filterData)
});
if (!response.ok) {
const message = await response.text()
throw new Error(message)
}

return await response //needs to be returned as raw not json
}

export default {
downloadTrainingCompletionReport
}

</script>
221 changes: 190 additions & 31 deletions training-front-end/src/components/TrainingReportDownload.vue
Original file line number Diff line number Diff line change
@@ -1,49 +1,208 @@
<script setup>
import { computed } from "vue"
import { useStore } from '@nanostores/vue'
import { profile} from '../stores/user'
import USWDSAlert from './USWDSAlert.vue'
import { computed, reactive, ref, watch} from "vue"
import { useStore } from '@nanostores/vue'
import { useVuelidate } from "@vuelidate/core";
import { profile } from '../stores/user'
import USWDSAlert from './USWDSAlert.vue'
import ValidatedDateRangePicker from "./form-components/ValidatedDateRangePicker.vue";
import ValidatedCheckboxGroup from "./form-components/ValidatedCheckboxGroup.vue";
import USWDSComboBox from "./form-components/USWDSComboBox.vue";
import {agencyList, bureauList, setSelectedAgencyId} from "../stores/agencies.js";
import ReportRepository from "./ReportRepository.vue";
import ReportUtilities from "./ReportUtilities.vue";
import SpinnerGraphic from "./SpinnerGraphic.vue";

const user = useStore(profile)
const isReportUser = computed(() => user.value.roles.includes('Report'))
const base_url = import.meta.env.PUBLIC_API_BASE_URL
const report_url = `${base_url}/api/v1/users/download-user-quiz-completion-report`
const error = ref()
const showSuccessMessage = ref(false)
const isLoading = ref(false)
const showSpinner = ref(false)
const user = useStore(profile)
const isReportUser = computed(() => user.value.roles.includes('Report'))
const agency_options = useStore(agencyList)
const bureaus = useStore(bureauList)
const report_agencies = user.value.report_agencies
let filteredAgencyOptions
let filteredBureauOptions

filteredAgencyOptions = computed(() => {
return agency_options.value.filter((agency) => report_agencies.map(x => x.name).includes(agency.name))
})

filteredBureauOptions = computed(() => {
return bureaus.value.filter((bureau) => report_agencies.map(x => x.id).includes(bureau.id))
})

//Properties
const user_input = reactive({
agency_id: undefined,
bureau_id: undefined,
quiz_names: undefined,
completion_date_range: undefined,
})

watch(() => user_input.agency_id, async() => {
setSelectedAgencyId(user_input.agency_id)
user_input.bureau_id = undefined
})

function setError(event){
error.value = event
}

const validation_info = {
agency_id: {},
bureau_id: {},
completion_date_range: {},
quiz_names: {}
}

const quiz_names_options = [
{value: 'Fleet Training For Program Coordinators', label: 'Fleet Training For Program Coordinators'},
{value: 'Purchase Training for Card/Account Holders and Approving Officials', label: 'Purchase Training for Card/Account Holders and Approving Officials'},
{value: 'Purchase Training For Program Coordinators', label: 'Purchase Training For Program Coordinators'},
{value: 'Travel Training for Agency/Organization Program Coordinators', label: 'Travel Training for Agency/Organization Program Coordinators'},
{value: 'Travel Training for Card/Account Holders and Approving Officials', label: 'Travel Training for Card/Account Holders and Approving Officials'},
];

const v_all_info$ = useVuelidate(validation_info, user_input)
Copy link
Contributor

Choose a reason for hiding this comment

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

useVuelidate by default probably can't handle completation_date_range date compare, maybe helpers also need to be imported to add customized date validation to handle start date greater than end date issue. Currently, if I put start date greater than end date and hit submit, it display "please ender a valid date" for end date, which is not correct, as "01/11/2022" is valid date.
image


//format dates from uswds standard to the format needed for backend
const formatDateToYYYYMMDD = (dates) => {
return dates ? dates.map(date => (date ? new Date(date).toISOString() : null)) : [];
};

async function downloadReport() {
isLoading.value = true
showSpinner.value = true
error.value = undefined
showSuccessMessage.value = false

try {
const dates = formatDateToYYYYMMDD(user_input.completion_date_range)
const model = {
'agency_id': user_input.agency_id || null,
'bureau_id': user_input.bureau_id || null,
'completion_date_start': dates[0] || null,
'completion_date_end': dates[1] || null,
'quiz_names': user_input.quiz_names || null
}
let response = await ReportRepository.downloadTrainingCompletionReport(model)
const blob = await response.blob();
await ReportUtilities.downloadBlobAsFile(blob, 'SmartPayTrainingReport.csv')
showSuccessMessage.value = true

} catch (error) {
setError(error)
}

isLoading.value = false
Copy link
Contributor

Choose a reason for hiding this comment

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

maybe isLoading and showSpinner reset can be in finally block instead to ensure isLoading and showSpinner are reset even exception happending?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Added reset of isLoading and showSpinner boolean values in a finally block

showSpinner.value = false
}

</script>
<template>
<section
v-if="isReportUser"
v-if="isReportUser"
class="usa-prose"
>
<h2>Download Your Report</h2>
<p>
We’ve created a report for you in CSV format. You can open it in the spreadsheet
application of your choice (e.g. Microsoft Excel, Google Sheets, Apple Numbers).
</p>
<form
:action="report_url"
method="post"
>
<input
type="hidden"
name="jwtToken"
:value="user.jwt"
>
<button
class="usa-button"
type="submit"
<div class="padding-top-4 padding-bottom-4 grid-container">
<h2>Download Your Report</h2>
<h2>Enter Report Parameters</h2>
<p>
The GSA SmartPay Training Report has no required parameters.
</p>
<p>
<b>Note:</b> If a report is generated with an individual completing multiple trainings, each training will be listed separately on the report.
</p>
<form
ref="form"
class="usa-form usa-form--large margin-bottom-3"
data-test="report-form"
@submit.prevent="downloadReport"
>
Download Report
</button>
</form>
<ValidatedDateRangePicker
v-model="user_input.completion_date_range"
client:load
:validator="v_all_info$.completion_date_range"
label="Completion date range"
name="Completion date range"
/>
<ValidatedCheckboxGroup
v-model="user_input.quiz_names"
:options="quiz_names_options"
:validator="v_all_info$.quiz_names"
name="Quiz type(s)"
legend="Quiz type(s)"
class="margin-top-4"
/>
<USWDSComboBox
v-model="user_input.agency_id"
client:load
:validator="v_all_info$.agency_id"
:items="filteredAgencyOptions"
label="Agency / organization"
name="Agency"
/>
<USWDSComboBox
v-if="filteredBureauOptions.length"
v-model="user_input.bureau_id"
client:load
:validator="v_all_info$.bureau_id"
:items="filteredBureauOptions"
label="Sub-agency, organization, or bureau"
name="Bureau"
/>
<input
class="usa-button"
type="submit"
value="Submit"
:disabled="isLoading"
data-test="submit"
>
<div>
<USWDSAlert
v-if="error"
status="error"
class="usa-alert--slim"
:has-heading="false"
>
{{ error }}
</USWDSAlert>
<USWDSAlert
v-if="showSuccessMessage"
status="success"
class="usa-alert--slim"
:has-heading="false"
>
Report has been generated.
</USWDSAlert>
</div>
<div class="grid-row grid-gap margin-top-3">
<!--display spinner along with submit button in one row for desktop-->
<div
v-if="showSpinner"
class="display-none tablet:display-block tablet:grid-col-1 tablet:padding-top-3 tablet:margin-left-neg-1"
>
<SpinnerGraphic />
</div>
</div>
<!--display spinner under submit button for mobile view-->
<div
v-if="showSpinner"
class="tablet:display-none margin-top-1 text-center"
>
<SpinnerGraphic />
</div>
</form>
</div>
</section>
<section v-else>
<USWDSAlert
<USWDSAlert
status="error"
class="usa-alert"
heading="You are not authorized to receive reports."
>
Your email account is not authorized to access training reports. If you should be authorized, you can
Your email account is not authorized to access training reports. If you should be authorized, you can
<a
class="usa-link"
href="mailto:[email protected]"
Expand Down
Loading
Loading