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

attempt to match optionId to value in case where optionId is not a uuid #91

Draft
wants to merge 3 commits into
base: master
Choose a base branch
from
Draft
Changes from all 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
70 changes: 62 additions & 8 deletions src/conditionalLogicService/evaluateConditionalOptionsPredicate.ts
Original file line number Diff line number Diff line change
@@ -3,23 +3,77 @@ import { FormTypes, ConditionTypes } from '@oneblink/types'
export default function evaluateConditionalOptionsPredicate({
predicate,
predicateElement,
predicateValue: value,
predicateValue,
}: {
predicate: ConditionTypes.ConditionalPredicateOptions
predicateValue: unknown
predicateElement: FormTypes.FormElementWithOptions
}): boolean {
/**
* In the @oneblink/apps function formService.parseFormElementOptions dynamic
* lists are parsed to ensure compliance with One Blink forms. If the dynamic
* list options include attributes and the element has attributesMapping
* defined then formService.parseFormElementOptions will attempt to match each
* dynamic list option attribute value with an option value in the predicate
* element's options using the element's attributesMapping. For each attribute
* in the element's attrbutesMapping If a predicate option is found for a
* particular attribute then the optionIds property of the predicate argument
* of this function will be appened with either the matched option's id or
* value.
*
* Eg
*
* const element = {
* ...elementProps,
* options: [
* {
* id: 'abc123',
* label: 'German Shepherd',
* value: 'germanShepherd',
* attributes: [{ elementId: 'xyz456', optionIds: ['aaa111'] }],
* },
* ],
* }
*
* If a predicate option is not found the optionIds property of the predicate
* argument will contain the dynamic list option's attribute value.
*
* Eg
*
* const element = {
* ...elementProps,
* options: [
* {
* id: 'abc123',
* label: 'German Shepherd',
* value: 'germanShepherd',
* attributes: [{ elementId: 'xyz456', optionIds: ['dog'] }],
* },
* ],
* }
*/
return predicate.optionIds.some((optionId) => {
const option = predicateElement.options?.find((o) => o.id === optionId)
if (option) {
if (Array.isArray(value)) {
return value.some((modelValue) => {
return modelValue === option.value

/**
* If the predicate element uses injected options using the {ELEMENT:Name}
* syntax we will not find an option. Referring to the second example above
* where the optionId is 'dog'. If an injected option's value parses to
* 'dog' then we can still match it. Ie. optionValue will be set to 'dog'
* and if the predicateValue is either 'dog' (radio, select, autocomplete)
* or ['dog'] (checkboxes, select) or {value:'dog'} (compliance) then we
* will return true for this predicate
*/
const optionValue = option?.value ?? optionId
if (optionValue) {
if (Array.isArray(predicateValue)) {
return predicateValue.some((modelValue) => {
return modelValue === optionValue
})
} else if (predicateElement.type === 'compliance' && value) {
return option.value === (value as { value: unknown }).value
} else if (predicateElement.type === 'compliance' && predicateValue) {
return optionValue === (predicateValue as { value: unknown }).value
} else {
return option.value === value
return optionValue === predicateValue
}
} else {
return false