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

Add additional interfaces for Scoped, Labelable and Labeled ui elements. #1935

Merged
merged 19 commits into from
Jun 21, 2022

Conversation

wienczny
Copy link
Contributor

Make Scopable scope optional and use Scoped with required scope where required.
Add type predicates for Scopeable, Scoped, Labelable and Labeled.
Duplicate old functions handling Scopeable to have Scopeable and Scoped where required.
Update toDataPathSegments to allow nullable schemaPath and return an empty array if this happens.

Make Scopable scope optional and use Scoped with required scope where required.
Add type predicates for Scopeable, Scoped, Labelable and Labeled.
Duplicate old functions handling Scopeable to have Scopeable and Scoped where required.
Update toDataPathSegments to allow nullable schemaPath and return an empty array if this happens.
@CLAassistant
Copy link

CLAassistant commented May 15, 2022

CLA assistant check
All committers have signed the CLA.

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.

Thanks for the pull request!

In general it definitely makes sense to me to improve our typings whenever possible.

Can you explain your motivation for this change? Especially refactoring a long standing existing interface like Scopable which could introduce backward compatibility problems.

Comment on lines 28 to 38
/**
* Interface for describing an UI schema element that may reference
* a subschema. The value of the scope may be a JSON Pointer or null.
*/
export interface Scopable {
/**
* The scope that determines to which part this element may be bound to.
*/
scope?: string;
}

Copy link
Member

Choose a reason for hiding this comment

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

If null is a foreseen value then this should be reflected in the typings, i.e. scope?: string | null. However we don't have any other nullebale types in JSON Forms, so I'm a bit hesitant to introduce this now. Is there a specific use case you have in mind?

This new interface with the optional scope is only used in one place: The new replacement for Scopable called Scoped. What is the use case of an element with optional scope?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

The case for this change would be the option to add scopes to existing classes without breaking the API in the future. This allows adding the interface to Categorization and Category in a second step to enable relative scopes in their children.

Comment on lines 262 to 263
export const isScopeable = (obj: object): obj is Scopable =>
obj !== undefined && obj.hasOwnProperty('scope');
Copy link
Member

Choose a reason for hiding this comment

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

This does not fit. In Scopable the scope is optional. So why check for it here?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

You are right it. The check is only useful if was

interface Scopable {
scope: string|null;
}

packages/core/src/models/uischema.ts Outdated Show resolved Hide resolved
packages/core/src/models/uischema.ts Outdated Show resolved Hide resolved
packages/core/src/models/uischema.ts Outdated Show resolved Hide resolved
packages/core/src/util/path.ts Outdated Show resolved Hide resolved
packages/core/src/models/uischema.ts Show resolved Hide resolved
@coveralls
Copy link

coveralls commented May 16, 2022

Coverage Status

Coverage remained the same at 84.098% when pulling 29e6d84 on wienczny:master into 31972a5 on eclipsesource:master.

wienczny and others added 2 commits May 16, 2022 12:36
toDataPathSegments: check for undefined, null and ''. null is not allowed by function definition, but it's included.
Remove unnecessary type guards and improve the remaining ones.
Copy link
Contributor Author

@wienczny wienczny left a comment

Choose a reason for hiding this comment

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

Did you have time to check my changes? Are there still open issues?

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 talked with the team. From our point of view it makes sense to support Scopable, Scoped, Labelable and Labeled. Including the backward compatibility break of changing the Scopable interface. We're shortly before the 3.0 release, so that's the perfect point to get this change still in.

If you clean up the PR to only include the interface changes and the type guards we're going to merge it 👍

packages/core/src/models/uischema.ts Show resolved Hide resolved
packages/core/src/models/uischema.ts Show resolved Hide resolved
packages/core/src/util/path.ts Outdated Show resolved Hide resolved
wienczny added 2 commits May 31, 2022 13:27
Replace Scopeable by Scoped where used
Add type guards for Scoped and Labeled
Make composeWithUI handle Scoped and Scopeable
@@ -249,7 +260,13 @@ export const isLayout = (uischema: UISchemaElement): uischema is Layout =>
(uischema as Layout).elements !== undefined;

export const isScopeable = (obj: object): obj is Scopable =>
obj !== undefined && obj.hasOwnProperty('scope') && (obj as Scopable).scope !== undefined;
obj !== undefined && obj.hasOwnProperty('scope');
Copy link
Member

Choose a reason for hiding this comment

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

scope is optional in Scopeable so it's allowed to be not set, therefore we should not check whether that property exists. Also we require the parameter to be an object so it can't be undefined.

I would like to suggest to improve the usefulness of the guard by marking the parameter as unknown and do some simple checks, i.e.

export const isScopeable = (obj: unknown): obj is Scopable =>
  obj && typeof obj === 'object';

This checks that obj is neither undefined nor null and it's something which we can assign scope to.

packages/core/src/models/uischema.ts Outdated Show resolved Hide resolved
Comment on lines 84 to 86
if (!isScoped(scopableUi)) {
return '';
}
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 to return '' here when later in the code we return path when the scope was empty.

The could could be clearer by

export const composeWithUi = (scopableUi: Scopable, path: string): string => {
  if (!isScoped(scopableUi)) {
    return path ?? '';
  }

  const segments = toDataPathSegments(scopableUi.scope);

  if (isEmpty(segments)) {
    return path ?? '';
  }

  return compose(path, segments.join('.'));
};

isScopeable(obj) && obj.scope !== undefined;

export const isLabelable = (obj: object): obj is Lableable =>
obj !== undefined && obj.hasOwnProperty('label');
Copy link
Member

Choose a reason for hiding this comment

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

This check should be adapted similarly to Scopable

wienczny and others added 10 commits June 2, 2022 11:17
use typeof obj.scope to check for Scoped

Co-authored-by: Stefan Dirix <sdirix@eclipsesource.com>
Make Scopable scope optional and use Scoped with required scope where required.
Add type predicates for Scopeable, Scoped, Labelable and Labeled.
Duplicate old functions handling Scopeable to have Scopeable and Scoped where required.
Update toDataPathSegments to allow nullable schemaPath and return an empty array if this happens.
toDataPathSegments: check for undefined, null and ''. null is not allowed by function definition, but it's included.
Remove unnecessary type guards and improve the remaining ones.
Replace Scopeable by Scoped where used
Add type guards for Scoped and Labeled
Make composeWithUI handle Scoped and Scopeable
use typeof obj.scope to check for Scoped

Co-authored-by: Stefan Dirix <sdirix@eclipsesource.com>
MIGRATION.md Outdated Show resolved Hide resolved
MIGRATION.md Show resolved Hide resolved
@sdirix sdirix merged commit ddd4315 into eclipsesource:master Jun 21, 2022
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.

4 participants