-
Notifications
You must be signed in to change notification settings - Fork 229
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
feat(#8681): support requiring countdown-timer fields #8826
feat(#8681): support requiring countdown-timer fields #8826
Conversation
09114f1
to
9d7c8fc
Compare
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.
This is a great start! I have added a few small suggestions for simplifying things, but then I noted one design element that we missed here! Once we make a decision on question-types and parameter vs default we can circle back for another round of changes (and look at adding some tests).
@jkuester Thanks a lot for the comments! working on the suggested changes. |
791293f
to
5b09e46
Compare
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.
Okay! Now that we finally landed on how we want to proceed, we can make some real progress here! I have added a couple suggestions in-line for how we can update the widget to support the trigger
questions. We will also need some changes to a few additional files:
medic.less
:
We need to update this line in the css to properly hide the trigger bullet elements. I think we can just change this line to be:
input, .option-wrapper {
Unit tests
Unfortunately, we do not have any existing unit tests for this widget, so we are going to have to add some. I think the best way to start would be to just try to follow the pattern in the phone-widget.spec.ts in a new countdown-widget.spec.ts
file. It would be great if you give this a shot! I am afraid these tests will get a bit tricky, tough, since we probably do not want to actually wait for the timer for 60sec
to complete. Hopefully we can stub around with sinon
and rewire
enough to get something useful. Please reach out, though if you get stuck!
I am going to work on adding the server-side code to the api that we need to actually load the cht:duration
attribute from the form xml. When I get that done, we can merge it in here and it can all go into master
together!
@SheilaAbby I got the server-side changes done! I put them all in this PR pointed back at your branch. When you are ready, you can merge that PR (into your |
That's good news! Thank you! Let me pull the changes |
6cf5335
to
e85a61f
Compare
e85a61f
to
4a41f87
Compare
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.
Alright @SheilaAbby! I have added some unit/e2e tests and everything seems to be working just like we want! I have just a few more minor cleanup comments here and I will also reach out to get a few more Medic colleagues to review what we have done.
# Conflicts: # tests/e2e/default/enketo/submit-countdown-timer-form.wdio-spec.js # tests/page-objects/default/enketo/countdown-timer.wdio.page.js # tests/page-objects/default/enketo/generic-form.wdio.page.js
@latin-panda @lorerod can you have a look at this PR when you get some time? Summary:The goal of this PR is to be able to require that a user run and wait for the timer, before continuing to the next page of the form. To support the Usage:
|
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.
Great contribution! I left some comments inline and also below:
-
@jkuester, is there any PR documenting this work in cht-docs? Something to include for example is: Why adding
cht=https://communityhealthtoolkit.org
in the setting tab? What other values are supported? -
You can also consider including a form with this @ support-scripts repo.
-
Testing:
Node: 16, 18 & 20
Phyton: 3.12
Form: enketo_widgets.xlsx see row 14
OS: Mac Sillicon
Verion: Latest from branchSheilaAbby:make-countdown-timer-required
Result: Not working for me
I get this error whenever I do the instructions here.

I tried changing the required
value from true()
to yes
like in some other xlsx but still not working. If I remove the widget in row 14, then it builds fine.
tests/e2e/default/enketo/submit-countdown-timer-form.wdio-spec.js
Outdated
Show resolved
Hide resolved
|
||
const audio = setupAudio(); | ||
|
||
const animate = function(canvas, duration, onComplete) { |
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.
Is it possible to put all the functions from here to the top (drawArc, drawCircle, etc.) inside the animate
function, so it works better as a "class"?
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.
So, I spent way longer than I should have agonizing over the best way to structure this code and I am still not at all happy with how it is. In general I am not a fan of nesting functions in other functions to create a pseudo-class. I think it is terrible for readability and maintainability since everything in the function has access to so much state and the logic flow can jump all over the place. This case is extra tricky since so much of the logic is tied to the mutable state of running
. From a readability perspective, I feel like a proper class here would be better than nested functions. However, JS classes are complete trash until ES2022 when they finally added support for private properties (while we are still targeting ES2018) and I could not find any other place in our entire code base where we actually used a JS class, so I was hesitant to introduce one here.
In the end I tried to split the difference and break out the functions that did not rely on the internal state of animate
so they could be considered separately while only keeping the internal functions in animate
that required access to the internal state of that function.... But like I said, I am still not really happy with how things turned out. Let me know your thoughts here! I am happy to just put all the functions back into animate
if you think that would be best!
@latin-panda thanks for the detailed feedback here! ❤️
|
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.
Thank you, @jkuester, for all the unit tests and e2e tests you added. You are even maintaining the deprecated functionality tested.
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.
It sparks ✨✨✨
Thanks!
Description
This update ensures the countdown-timer widget applies to questions with the input type 'trigger' rather than 'note'.
In contrast to a note, a 'trigger' input can be configured with 'required' set to 'yes'. This setup ensures that if a user tries to bypass the timer before it completes, they will get a
'This field is required'
constrain message.NB: Just like the previous implementation on a note, workflow developers are allowed set custom timer durations, now on a new column labelled
instance::cht:duration
Here is a sample xlsx that was used to test this feature
Uploading assessment_endemic.xlsx…
The updated countdown widget code will identify the trigger question types with the attached countdown timer and set the trigger question type OK radio button as checked when the timer completes running. The selected OK value can then be used to control subsequent questions linked to the countdown widget.
See screenshot,

#8681
Code review checklist
Compose URLs
If Build CI hasn't passed, these may 404:
License
The software is provided under AGPL-3.0. Contributions to this project are accepted under the same license.