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

Mixed support #2409

Open
wants to merge 42 commits into
base: master
Choose a base branch
from
Open

Conversation

kchobantonov
Copy link
Contributor

@kchobantonov kchobantonov commented Dec 30, 2024

  1. provide support for mixed types
  2. fix clearing of inputs when not under an object - e.g. string once cleared will become empty string rather than to return undefined to remove the property from the parent object
  3. fix date controls, now the date can show year, year/month and etc. depending on the control option views
  4. improve the performance to convert dayjs format into maska format function
  5. fixing the :key to at least have the collection size into the key so when the size changes the vue will not reuse html elements leaving attributes like id unchanged
  6. add more examples to show how the mixed control can be used
  7. use arraySchema from the mapped values rather than using Resolve
  8. minor UI changes to use less space
  9. core changes : use the provided schema when the resolve failed, make sure that boolean is valid schema as well - e.g. items: true is valid schema (Note: the last change for the items to be boolean was reverted back since more code should be adjusted - maybe a future improvement)

…ot part of the object that has the property defined in properties then to not clear it - e.g. send undefined value
… based on index - e.g. prevent reusing elements when the array change size which will cause id properties for example to not have correct value
…n also allow such case to be handled by the object renderer
…s no need to resolve that schema just use it
Copy link

netlify bot commented Dec 30, 2024

Deploy Preview for jsonforms-examples ready!

Name Link
🔨 Latest commit 6b863d9
🔍 Latest deploy log https://app.netlify.com/sites/jsonforms-examples/deploys/67e144a955ab9f00083c6e4d
😎 Deploy Preview https://deploy-preview-2409--jsonforms-examples.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

@kchobantonov
Copy link
Contributor Author

@coveralls
Copy link

coveralls commented Jan 7, 2025

Coverage Status

coverage: 82.453% (-0.06%) from 82.517%
when pulling 6b863d9 on kchobantonov:mixed-support
into 0628b99 on eclipsesource:master.

@sdirix
Copy link
Member

sdirix commented Jan 10, 2025

Thanks for the contribution ❤️ I'll take a look within January 👍

Copy link
Member

@sdirix sdirix left a comment

Choose a reason for hiding this comment

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

Hi @kchobantonov,

I had a first look at the implementation. The examples work great and are really impressive ❤️

I added some comments, please have a look

Comment on lines +126 to +132
if (
typeof schema === 'boolean' &&
(!pathSegments || pathSegments.length === 0)
) {
// add the case where the schema can be true value for example: items: true or additionalProperties: false
return schema;
}
Copy link
Member

Choose a reason for hiding this comment

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

What is the use case of this? Who hands over segments to resolve if the schema is just a boolean already anyway?

Copy link
Contributor Author

@kchobantonov kchobantonov Feb 13, 2025

Choose a reason for hiding this comment

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

that is to resolve the case where the value of the last path segment is equals to true. For example items: true and you want to bind that items then you need to resolve the jsonschema which can be boolean but in the code we ignore that this can be ever the case.

Also the next check for isEmpty if we pass true/false values which are valid JSON schema will return an undefined when resolved instead of boolean

Copy link
Member

Choose a reason for hiding this comment

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

The case is already handled via

  if (!pathSegments || pathSegments.length === 0) {
    return schema;
  }

If having a boolean schema breaks the checks before that, then we should fix those checks instead of hard coding the exception at the top.

I guess the typing does not properly reflect that the schema could be a boolean. So we should adjust the typing and the checks to handle the typing.

@@ -20,7 +20,7 @@
/>
<dispatch-renderer
v-for="(allOfRenderInfo, allOfIndex) in allOfRenderInfos"
:key="`${control.path}-${allOfIndex}`"
:key="`${control.path}-${allOfRenderInfos.length}-${allOfIndex}`"
Copy link
Member

Choose a reason for hiding this comment

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

Why adapt this key?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

because if the size of the allOfRendererInfo changes then the key is not stable enough to guarantee that previous keys will point the exact same elements - e.g. we do not have a primary key for the info that we can use - usually this happens when we change the UI schema and the schema have to be applied again.

Copy link
Member

Choose a reason for hiding this comment

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

Seems weird as this only fixes the case in which the uischema or schema changes and the length ALSO changes. It will not work if the length does not change.

I wonder whether there should be a change higher up, for example setting a key on the whole form and increasing it whenever we hand over a new schema or uischema. This would be a more generic fix covering all use cases, instead of only a partial fix, covering some of the use cases.

@kchobantonov
Copy link
Contributor Author

@sdirix check my responses and also updated repo but it looks like the build failed perhaps because netlify is using greater than pnpm version 8 now ?

Copy link
Member

@sdirix sdirix left a comment

Choose a reason for hiding this comment

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

I have concerns about the proposed changes to the core functionality.

Please have a look at my new comments and the unresolved comments from my previous review.

Comment on lines +876 to +889
if ((resolvedSchema as any) === true) {
// help the testers to determine the mixed control
resolvedSchema = {
type: [
'array',
'boolean',
'integer',
'null',
'number',
'object',
'string',
],
};
}
Copy link
Member

Choose a reason for hiding this comment

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

I don't like this. Up until now, the schema returned from the mappers was always a sub-schema of the overall schema. Not only does this not return an actual sub-schema, it also produces a new resolvedSchema whenever it is executed, which will break memo barriers.

I'm all for being able to handle boolean (sub-)schemas but the change should be less intrusive. I would like to keep the original prerequisite for now: Always return a sub-schema of the original schema. Once we break that, for example to support schema overrides, this becomes much more complicated because of the object identities.

@@ -871,7 +872,21 @@ export const mapStateToArrayControlProps = (
const { path, schema, uischema, label, ...props } =
mapStateToControlWithDetailProps(state, ownProps);

const resolvedSchema = Resolve.schema(schema, 'items', props.rootSchema);
let resolvedSchema = Resolve.schema(schema, 'items', props.rootSchema);
if ((resolvedSchema as any) === true) {
Copy link
Member

Choose a reason for hiding this comment

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

I don't like the cast to any here. If we really expect Resolve.schema to being able to return booleans, then its typing should indicate that.

ownProps.schema || rootSchema,
controlElement.scope,
rootSchema
) ?? ownProps.schema; // if resolve fails then rely that the ownProps.schema if exist
Copy link
Member

Choose a reason for hiding this comment

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

I feel this is plastering over bugs. If a schema can't be resolved then this means the handed over schema, uischema combination was wrong. It's better to fail early than falling back to the unresolved schema.

Comment on lines +126 to +132
if (
typeof schema === 'boolean' &&
(!pathSegments || pathSegments.length === 0)
) {
// add the case where the schema can be true value for example: items: true or additionalProperties: false
return schema;
}
Copy link
Member

Choose a reason for hiding this comment

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

The case is already handled via

  if (!pathSegments || pathSegments.length === 0) {
    return schema;
  }

If having a boolean schema breaks the checks before that, then we should fix those checks instead of hard coding the exception at the top.

I guess the typing does not properly reflect that the schema could be a boolean. So we should adjust the typing and the checks to handle the typing.

@@ -20,7 +20,7 @@
/>
<dispatch-renderer
v-for="(allOfRenderInfo, allOfIndex) in allOfRenderInfos"
:key="`${control.path}-${allOfIndex}`"
:key="`${control.path}-${allOfRenderInfos.length}-${allOfIndex}`"
Copy link
Member

Choose a reason for hiding this comment

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

Seems weird as this only fixes the case in which the uischema or schema changes and the length ALSO changes. It will not work if the length does not change.

I wonder whether there should be a change higher up, for example setting a key on the whole form and increasing it whenever we hand over a new schema or uischema. This would be a more generic fix covering all use cases, instead of only a partial fix, covering some of the use cases.

context: TesterContext,
) =>
isMixedSchema(uischema, schema, context) &&
(isControl(uischema) || isDefaultGenUiSchema(uischema));
Copy link
Member

Choose a reason for hiding this comment

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

Why do we check isDefaultGenUiSchema? Note also that we have adopters who adopt the default UI Schema, so this might not apply to them.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants