Skip to content

Commit

Permalink
feat: Filter Storage Mode Responses by Submission Id (#174)
Browse files Browse the repository at this point in the history
* feat: Filter Storage Mode Responses by Submission Id (updated 2020-09-13)

* Update page design so that it is compatible with tablets and everything is on one row

* Update CSS for iPad
  • Loading branch information
frankchn authored and tshuli committed Sep 22, 2020
1 parent 6147599 commit e8823e4
Show file tree
Hide file tree
Showing 6 changed files with 142 additions and 16 deletions.
44 changes: 39 additions & 5 deletions src/app/controllers/encrypt-submissions.server.controller.js
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,8 @@ const {
getProcessedResponses,
} = require('../modules/submission/submission.service')

const HttpStatus = require('http-status-codes')

/**
* Extracts relevant fields, injects questions, verifies visibility of field and validates answers
* to produce req.body.parsedResponses
Expand Down Expand Up @@ -223,15 +225,47 @@ exports.saveResponseToDb = function (req, res, next) {
*/
exports.getMetadata = function (req, res) {
let pageSize = 10
let { page } = req.query || {}
let { page, submissionId } = req.query || {}
let numToSkip = parseInt(page - 1 || 0) * pageSize

let matchClause = {
form: req.form._id,
submissionType: 'encryptSubmission',
}

if (submissionId) {
if (mongoose.Types.ObjectId.isValid(submissionId)) {
matchClause._id = mongoose.Types.ObjectId(submissionId)
// Reading from primary to avoid any contention issues with bulk queries on secondary servers
Submission.findOne(matchClause, { created: 1 })
.read('primary')
.exec((err, result) => {
if (err) {
logger.error(getRequestIp(req), req.url, req.headers, err)
return res.status(HttpStatus.INTERNAL_SERVER_ERROR).send({
message: errorHandler.getMongoErrorMessage(err),
})
}
if (!result) {
return res.status(HttpStatus.OK).send({ metadata: [], count: 0 })
}
let entry = {
number: 1,
refNo: result._id,
submissionTime: moment(result.created)
.tz('Asia/Singapore')
.format('Do MMM YYYY, h:mm:ss a'),
}
return res.status(HttpStatus.OK).send({ metadata: [entry], count: 1 })
})
} else {
return res.status(HttpStatus.OK).send({ metadata: [], count: 0 })
}
}

Submission.aggregate([
{
$match: {
form: req.form._id,
submissionType: 'encryptSubmission',
},
$match: matchClause,
},
{
$sort: { created: -1 },
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,12 @@ function ViewResponsesController(
vm.isEncryptResponseMode = vm.myform.responseMode === responseModeEnum.ENCRYPT
vm.encryptionKey = null // will be set to an instance of EncryptionKey when form is unlocked successfully
vm.csvDownloading = false // whether CSV export is in progress

vm.attachmentDownloadUrls = new Map()
vm.filterBySubmissionRefId = '' // whether to filter submissions by a specific ID
vm.filterBySubmissionRefIdTextbox = ''
vm.filterBySubmissionRefIdMatcher = /^[0-9A-Fa-f]{24}$/
vm.filterBySubmissionShowFilterBox = false

// Three views:
// 1 - Unlock view for verifying form password
Expand Down Expand Up @@ -318,12 +323,29 @@ function ViewResponsesController(
}
})

vm.filterBySubmissionChanged = function () {
// We only reload the table if the text box has changed. This prevents excessive
// requests being sent by users clicking on the "Filter" button repeatedly.
if (
vm.filterBySubmissionRefIdTextbox !== '' &&
vm.filterBySubmissionRefId !== vm.filterBySubmissionRefIdTextbox
) {
vm.filterBySubmissionRefId = vm.filterBySubmissionRefIdTextbox
vm.tableParams.reload()
}
}

vm.filterBySubmissionReset = function () {
vm.filterBySubmissionShowFilterBox = false
vm.filterBySubmissionRefId = ''
vm.filterBySubmissionRefIdTextbox = ''
vm.tableParams.reload()
}

// Called by child directive unlockResponsesForm after key is verified to get responses
vm.loadResponses = function (formPassword) {
vm.formPassword = formPassword
vm.loadResponses = function () {
vm.currentView = 2
vm.loading = true

vm.tableParams = new NgTableParams(
{
page: 1, // show first page
Expand All @@ -334,6 +356,7 @@ function ViewResponsesController(
let { page } = params.url()
return Submissions.getMetadata({
formId: vm.myform._id,
filterBySubmissionRefId: vm.filterBySubmissionRefId,
page,
})
.then((data) => {
Expand Down
1 change: 0 additions & 1 deletion src/public/modules/forms/admin/css/export-button.css
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@
display: flex;
justify-content: center;
align-items: center;
min-width: 164px;
margin-left: auto;
}

Expand Down
22 changes: 21 additions & 1 deletion src/public/modules/forms/admin/css/view-responses.css
Original file line number Diff line number Diff line change
Expand Up @@ -101,7 +101,8 @@
}

#responses-tab .response-stats {
width: 100%;
white-space: nowrap;
min-width: 240px;
}

#responses-tab .datepicker-export-container {
Expand Down Expand Up @@ -715,3 +716,22 @@
padding-left: 16px !important;
border: solid 1px #ccc;
}

#filter-by-submission-reset-link {
padding-top: 13px;
padding-left: 13px;
white-space: nowrap;
cursor: pointer;
}

#filter-by-submission-filter-link {
padding-top: 13px;
padding-left: 5px;
white-space: nowrap;
cursor: pointer;
}

#filter-by-submission-filter-textbox-container {
display: flex;
min-width: 240px;
}
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,9 @@

<!-- Storage Mode -->
<div ng-if="!vm.loading && vm.isEncryptResponseMode">
<div ng-if="vm.responsesCount === 0">
<div
ng-if="vm.currentView === 1 && vm.responsesCount === 0 && vm.filterBySubmissionRefId === ''"
>
<div class="flex-column">
<img
id="no-responses"
Expand Down Expand Up @@ -62,17 +64,43 @@
>
</verify-secret-key-directive>
</div>
<div ng-if="vm.responsesCount > 0 && vm.currentView === 2">
<div ng-if="vm.currentView === 2">
<div class="flex-row">
<div class="response-stats">
<div class="col-md-3 response-stats">
<span class="stats-text">
<span
><span class="stats">{{vm.responsesCount}}</span> response(s) to
date</span
>
<a
id="filter-by-submission-filter-link"
ng-show="!vm.filterBySubmissionShowFilterBox"
ng-click="vm.filterBySubmissionShowFilterBox = true"
>
<i class="bx bx-search"></i>
</a>
</span>
</div>
<div
class="col-md-6"
ng-show="vm.filterBySubmissionShowFilterBox"
id="filter-by-submission-filter-textbox-container"
>
<input
class="input-custom input-medium"
ng-model="vm.filterBySubmissionRefIdTextbox"
ng-change="vm.filterBySubmissionChanged()"
ng-model-options="{ debounce: 150 }"
placeholder="Search by reference no."
/>
<span
id="filter-by-submission-reset-link"
ng-show="vm.filterBySubmissionRefId !== ''"
>
<a ng-click="vm.filterBySubmissionReset()">View&nbsp;All</a>
</span>
</div>
<div class="datepicker-export-container">
<div class="datepicker-export-container col-md-3">
<date-range-picker-directive
class="datepicker-container"
ng-model="vm.datePicker.date"
Expand All @@ -84,8 +112,11 @@
></export-button-component>
</div>
</div>
<div class="flex-row" ng-show="vm.filterBySubmissionShowFilterBox">
<div class="col-md-5"></div>
</div>
<div class="row">
<div class="col-md-12">
<div ng-if="vm.responsesCount > 0" class="col-md-12">
<table class="table" ng-table="vm.tableParams" show-filter="false">
<tr ng-repeat="row in $data" ng-click="vm.rowOnClick($index)">
<td
Expand All @@ -108,6 +139,21 @@
</tr>
</table>
</div>
<div
ng-if="vm.responsesCount === 0 && vm.filterBySubmissionRefId !== ''"
>
<div class="flex-column">
<img
ng-src="/public/modules/core/img/error-illustration.svg"
id="no-responses"
/>
<div class="title">No results found</div>
<div class="subtitle">
Did you enter the right reference number? We can't seem to find
the response.
</div>
</div>
</div>
</div>
</div>
<div
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -146,10 +146,14 @@ function SubmissionsFactory(
},
getMetadata: function (params) {
const deferred = $q.defer()
const resUrl = `${fixParamsToUrl(params, submitAdminUrl)}/metadata?page=${
let resUrl = `${fixParamsToUrl(params, submitAdminUrl)}/metadata?page=${
params.page
}`

if (params.filterBySubmissionRefId) {
resUrl += `&submissionId=${params.filterBySubmissionRefId}`
}

$http.get(resUrl).then(
function (response) {
deferred.resolve(response.data)
Expand Down

0 comments on commit e8823e4

Please sign in to comment.