-
Notifications
You must be signed in to change notification settings - Fork 652
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
[WIP] oneOf/anyOf support #505
base: development
Are you sure you want to change the base?
Conversation
…chemas Add some other utility functions to help manipulate schemas
Hey @gazpachoking great that you put it out there 👍 This is on the roadmap for us pretty soon and I've already done some proof of concept on it myself. I won't merge directly, but as soon as the new builder is up and running I'll take a look at it! |
@davidlgj Good to hear. How is your proof of concept going about it (that's all this pr is at this point too)? Just trying to figure out if I should continue with this, or if there is some sort of better way to structure a solution that you are already planning. I spent most of my time just reading the code so far, and the method I've implemented seemed like the path of least resistance, though I'm not sure if it's actually a good one. |
Honestly I can't remember! I did it before going on vacation for the summer :) I'll try to find the branch later today. But I think I must be doing something similar. |
Change model when form is changed
Ok, so I found my code I pushed it here: https://github.com/Textalk/angular-schema-form/tree/feature/something-of-other and relevant commit with diff here 9d58aca I taken a similar approach but managed without a directive, only did anyOf, with the specific case the you have a no type and just a anyOf. It's a proof of concept I'm lacking the extending of the schemas though. One of the things I was aiming at was making the form definition similar to what arrays has. I.e. you need to be able to have a form def that specifies that the input in this version of the anyof should have this placeholder, or have this type etc. So for that I use a key notation with var schema = {
type: 'object',
properties: {
subform: {
title: 'Subform',
anyOf: [
{
type: 'object',
properties: {
name: {type: string}
comment: {type: string}
}
},
{
type: 'object',
properties: {
name: {type: string}
email: {type: string}
}
}
]
}
}
};
var form = [
{
key: 'subform',
children: [ // named "items" in my poc
[
'subform{0}.name',
{
key: 'subform{0}.comment',
type: 'textarea'
}
],
[
'subform{1}.name',
{
key: 'subform{1}.email',
type: 'text',
pattern: '^\S+@\S+$'
}
]
]
}
]; As I said before I need to get the new builder up and running (almost there, mostly testing and docs left), but we should be looking at anyOf support soon. Please keep me updated if you make any further efforts on it! |
Yeah, we have similar things going on. I'm just using the directive to do stuff like switch the form based on the model data, and switch the model data based on the selected form. I'll take a closer peek at your ideas and see what sort of modifications would be good. I was also struggling with how to manually specify the form definitions for the different schemas, I'll see what I can come up with. |
Do we really need the subform name or number in the key though? That form will only be used when selected, and the key is normally pointing to the position in the model, not the position in the form right? With arrays it's a bit different, as we are creating forms for different model locations, (the array indexes,) whereas here all the forms are all for the same model location. |
Hmmm.... you might be right! If the index in the form definition sort of specifies which subform it belongs to, like this: var form = [
{
key: 'subform',
children: [
[ // Since this is index 0 we now it's index zero in the anyOf
'subform.name',
{
key: 'subform.comment',
type: 'textarea'
}
],
[ // Since this is index 1 we now it's index one in the anyOf
'subform.name',
{
key: 'subform.email',
type: 'text',
pattern: '^\S+@\S+$'
}
]
]
}
];```
btw the bootstrap decorator in the development branch will soon be deprecated in favor of the new one here https://github.com/Textalk/angular-schema-form-bootstrap
Just so you know :) |
Yep, index order is what I was going with, but I currently don't allow the array definition of a form. (using 'items' rather than 'children' as well atm) Am I right in thinking that the array type form definition is just a shortcut for a 'fieldset' type with that array as the 'items' key? |
any updates or progress on this? |
@davidlgj are you planning on using this brach or writing your own oneOf/anyOf support? I'd like to start work on this PR as the project i'm working one requires this support but i didn't want to waste effort if you are working on a different branch? |
Alright @gazpachoking and @stevezau. Feels like we need to get this done :) |
@davidlgj Awesome, sounds good 😄 |
@davidlgj great! I'm keen to get this completed! |
@gazpachoking @stevezau @DanielSchaffer Hi all! I've merged this PR @gazpachoking into https://github.com/Textalk/angular-schema-form/tree/gazpachoking-oneOf and the template and builder part into https://github.com/Textalk/angular-schema-form-bootstrap/tree/feature/anyof-or-bust I'm thinking any further development should happen in these branches. And I love to get your help on this 😄 To test it, check out both branches. Build them both (gulp watch is nice during development). Symlink The first form, a anyOf with to different types seems to work somewhat, but the second, an array with anyOf in items does not. So I also merged in some things from my own stab at anyOf support. Let me explain some things. Let me explain why this is needed. ASF works like this. So one property in the schema that is an So you see we have a problem with the lookup, since all n+1 form objects would get the same key! ( I think this is at least necessary internally, but maybe not needed in the user supplied form definition if we're smart in the "merge" function here: https://github.com/Textalk/angular-schema-form/blob/gazpachoking-oneOf/src/services/schema-form.js#L434 We can probably check for "formselect" and then add proper Stripping key of I also changed from ng-show to ng-if since if the subform is in the DOM it will be validated. Ok. So what's left? A lot.
|
Trying to get my environment set up properly (all these js tools are new to me, npm, bower, gulp.) Then I'll try to grok the changes and see what I'm thinking. |
I've submitted a PRs to each of the branches to fix a couple things, including the dists not including all the changes. Once that was all settled (and I changed my local angular-schema-form-bootstrap bower config fork to look at my fixed angular-schema-form repo), I was able to load the example correctly. However, this code only seems to work when switching between different properties, but not when you want to switch between entirely different schemas - for example, a "person" schema: {
"type": "object",
"title": "person",
"properties": {
"name": {
"type": "string"
},
"age": {
"type": "number"
},
"primaryTransportation": {
"type": "object",
"oneOf": [{
"title": "pedestrian",
"properties": {
"type": {
"type": "string",
"enum":["pedestrian"]
}
}
}, {
"title": "automobile",
"properties": {
"type": {
"type": "string",
"enum": ["bicycle", "moped", "motorcycle", "car", "truck", "suv", "van"]
},
"make": {
"type": "string"
},
"model": {
"type": "string"
}
}
}, {
"title": "public transit",
"properties": {
"type": {
"type": "string",
"enum": ["bus", "subway", "train", "other light rail", "shuttle"]
},
"route": {
"type": "string"
}
}
}]
}
}
} Given the above, I'd expect to see:
I'm still pretty new to JSON schema, so please let me know if this is completely off-base - but it's the pattern that I'm needing to support for my app. I was actually able to get it to generate the form using an add-on. It doesn't work alongside the changes you guys made, but here's what I've got so far: // someOf.js
angular.module('schemaForm').config(['schemaFormDecoratorsProvider', 'schemaFormProvider', 'sfBuilderProvider', 'sfPathProvider', function (
decoratorsProvider, sfProvider, sfBuilderProvider, sfPathProvider) {
var ofNodeTypes = {
oneOf: 'oneof',
anyOf: 'anyof',
allOf: 'allof'
};
function someOfSelector(name, schema, options) {
if (!schema[schema.type] || !ofNodeTypes[schema.type]) {
return null;
}
var selectorPath = options.path.slice();
selectorPath.push('selected');
var f = sfProvider.stdFormObj(name, schema, options);
f.key = selectorPath;
f.type = ofNodeTypes[schema.type];
f.titleMap = _.chain(schema[schema.type])
.map(function (item) {
return [item.title || item.name, item.title || item.name];
})
.object()
.value();
f.selector = f;
options.lookup[sfPathProvider.stringify(selectorPath)] = f;
return f;
}
function someOf(name, schema, options) {
var node = _.first(_.intersection(_.keys(ofNodeTypes), _.keys(schema)));
if (!node) {
return null;
}
var fieldset = sfProvider.defaultFormDefinition(name, _.omit(schema, node), options);
if (!fieldset) {
debug.warn('no fieldset!');
return null;
}
var selector = sfProvider.defaultFormDefinition(node, _.extend(_.pick(schema, node, 'title', 'name'), { type: node }), options);
if (!selector) {
debug.warn('no selector!');
return null;
}
fieldset.selector = selector;
fieldset.items.push(selector);
angular.forEach(schema[node], function (item) {
var optionForm = sfProvider.defaultFormDefinition(item.title || item.name, angular.extend(item, { type: 'object' }), options);
if (optionForm) {
optionForm.selector = selector;
optionForm.condition = 'form.selector.current === \'' + (item.title || item.name) + '\'';
fieldset.items.push(optionForm);
}
});
return fieldset;
}
var simpleTransclusion = sfBuilderProvider.builders.simpleTransclusion;
var ngModelOptions = sfBuilderProvider.builders.ngModelOptions;
var ngModel = sfBuilderProvider.builders.ngModel;
var sfField = sfBuilderProvider.builders.sfField;
var condition = sfBuilderProvider.builders.condition;
var array = sfBuilderProvider.builders.array;
sfProvider.prependRule('object', someOf);
_.each(ofNodeTypes, function (type, node) {
sfProvider.prependRule(node, someOfSelector);
decoratorsProvider.defineAddOn('bootstrapDecorator', type, 'view/decorators/bootstrap/someof.html', [
sfField, ngModel, ngModelOptions, condition
]);
});
}]); <!-- view/decorators/bootstrap/someof.html -->
<div ng-class="{'has-error': form.disableErrorState !== true && hasError(), 'has-success': form.disableSuccessState !== true && hasSuccess(), 'has-feedback': form.feedback !== false}" class="form-group {{form.htmlClass}} schema-form-select"><div>form.selector {{form.selector.current}}</div><label ng-show="showTitle()" class="control-label {{form.labelHtmlClass}}">{{form.title}}</label><select name="{{form.key.slice(-1)[0]}}" ng-model="form.selector.current" ng-disabled="form.readonly" ng-options="item.value as item.name group by item.group for item in form.titleMap" class="form-control {{form.fieldHtmlClass}}"></select><div sf-message="form.description" class="help-block"></div></div> The way it works is that it adds a new rule for the "object" type which creates a new fieldset, schema selector, and subforms when it encounters an object that has one of the "someOf" properties (oneOf/anyOf/allOf). There are also new form type rules added specifically for the "someOf" types which are responsible for generating the aforementioned schema selector (see someOfSelector). The decorator is basically a modified version of the existing "select" bootstrap decorator. What I like about how all this works is that it was still based on all the existing functionality - so regardless of what structure the schemas for the "someOf" property are, it is still able to recurse into the structure and generate all of the required subforms.
In order to solve the issue of storing the "selected" state in order to show the correct form, I'm sort of abusing JS object references - I put a reference to the selector form on each of the fieldset, subforms, and the selector itself so they can all look at the same thing, and then the selector's ng-model is bound to I think my solution is also sort of predicated on the "type" property pattern I have in my above example
What this doesn't cover yet is preselecting the right schema, and I'm also not sure that anyOf/oneOf/allOf validation is working correctly, though the individual fields in the subform to get validated. I'm sure I'm misusing stuff with this solution though, so if that's the case, feel free to point it out. Updated: Made some changes in my add-on code and updated the description to reflect it. |
I put my code in a new repo: https://github.com/DanielSchaffer/angular-schema-form-someof It now has a solution for preselecting the correct schema given a model - see the sfSomeOfSelector and sfSomeOf directives. Again, I feel like I'm doing some odd contortions with object references and the form object in order to accomplish these things, so feel free to point out if anything is particularly bad. But this appears to be working for the usage pattern I described above. Let me know what you think! |
@davidlgj I made a couple PRs to your branches which fixes the example at least a bit. I think the reason I wasn't using ng-if, is because when that bit of form gets destroyed, the model value gets deleted as well, which is I believe why the directive isn't working currently. I'll have a look at that next. @DanielSchaffer It's hard to evaluate your changes, since it isn't a fork of this repo, so I can't make github give me a nice diff. |
@gazpachoking it wouldn't be a diff since it's an addon - it's all new code. That said, it is dependent on changes to asf that I did do in a fork (https://github.com/DanielSchaffer/angular-schema-form/tree/danielschaffer-someof) |
Ahh. Guess I didn't read your stuff carefully enough, I'll have another look through. |
@gazpachoking I merged your PR's and I like that you thought of the example with a formselect used without an anyOf in the schema! @DanielSchaffer I just skimmed the code of your add-on, I'll try to take a deeper look later. But I did add the example schema you used in a previous comment: #505 (comment) And after @gazpachoking fixes it seems to render properly! |
@gazpachoking Just wanted to mention a thing I saw but have no immediate fix for: So here we like to validate the sub form. Sadly the sfValidator does not handle validating a chunk of a form, it just validates a form field. So we get an exception here on fieldsets for instance. So one way to fix this is to wrap each subform in an We could also drop down to using tv4 directly on each subform and it's model. This could be very nice if we use But this leaves us in a different pickle since we then don't take into account fields that are in the schema but not in the form definition! So if we use the tv4 method we need to filter out errors not in the subform (doable I guess...). A third solution might be recursively traverse items and call validate on each field that has some sort of validation. This might be the cleanest solution... |
@gazpachoking do you have any response to @davidlgj 's last post? |
So I followed @davidlgj:s instructions above and it worked, to my amazeballsment. WRT to todo list: 3 . Select the first subform that validates (not counting required errors) as the selected form. 5 . Should we clear fields that get validation errors when changing subform? 7 . Make it look more pretty? |
Preface: I haven't touched the code in a while, so my input is based on memory. I also haven't had much motivation/free time alignment for this recently, but I'll at least comment.
I don't think this breaks any current functionality, the display components will just get the edited schema, dynamically merged/edited from the input schemas. As for dynamically merging schemas, I think ultimately, it's what any solution needs to do. The display componentry needs one set of things to know what it should display. This choice just keeps the merged format still expressed in json schema. Perhaps there is a better place to do it though, not sure.
👍
I think right now it's storing the data separately for each form choice (or at least that was my goal,) such that when you go back to the same choice, you have the last data you entered for that anyOf/oneOf selection. Whether it can try to intelligently transfer the data as the forms are changed I'm still not sure. |
I have used it for a couple of days now, and I see just one problem with adding this like it is, and that is automatically selecting the first validating type, which I suppose must be done for it to be editable. But even that could be left in, actually, in my view this is about being able to start development of projects that use this functionality, if that is in the next minor, that will do it at least for me, for now. Neither of the other issues with it that have been brought up is a show stopper in my view. They can all be seen as additional and non-breaking functionality, and as such they can be added in subsequent releases. It might even be a good idea to let them ferment for a while. WRT to copying the schema, it breaks in a 1.0, and only really ugly hacks, so I see no huge issues with that either. |
@gazpachoking WRT to dynamically changing the schema, I am all for the anyOf functionality doing it in its own copy of it. What I see as ugly is for users doing it in their ASF implementations. If that makes sense. |
is there any news on when anyOf / oneOf / $ref will appear? these missing features are keeping me from using this library :( |
Any news on this ? |
interested too in the progress of support for anyOf / oneOf / $ref |
Hi, I'd like to start working on this but this seems quite behind master. where should I start from? Are https://github.com/Textalk/angular-schema-form/tree/gazpachoking-oneOf and https://github.com/Textalk/angular-schema-form-bootstrap/tree/feature/anyof-or-bust the most up-to-date branches ? |
@Vaevictus @MGouaillier @cvietor no the new webpack/babel version is progressing, but this feature hasn't been added to it yet. @mathroc those look like the latest to me, would need to be migrated to be a branch off webpack/babel in ASF and the development branch of json-schema-form-core ideally as they are the next major release of ASF, if you need a run through of it or have any questions let me know (keeping in mind I'm in Australian EST time) |
thanx for the input @Anthropic I'm gonna take a look at all that! I'll be trying to replace the current version with webpack/babel branch in an existing project. And if I can make it working I'll try to add this feature in json-schema-form-core and the webpack/babel branch |
another question: are angular-schema-form-bootstrap & angular-schema-form-material supposed to be working with the new branch too ? or should I expect to make adjustment regarding the webpack branch ? |
@mathroc the decorators should work, I have them working locally. The webpack/Babel branch in theory should work ok for everything except for ace textarea and colour picker. Everything else appears to be working, but I would not consider it ready for production as we still need more unit tests and to add back support for the items above while also testing and fixing array behaviour. |
Hi guys, is this still moving? We'd really like to use the library but this is a must have feature for our application. |
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.
I have used this for a while now, and I have not had any problems. I am sure there are minor issues left in there, but nothing that stops me from using it.
@nicklasb based on @davidlgj 's comment above #505 (comment) did you solve that for yourself or just not run into it? |
I will send 50 bucks to who ever can merge or fork this to the latest version |
Happy New Year! Let's get this going for real. |
Happy new year. Whats the plan regarding upgrading to ng2+ ? |
What's the holdup here? Anything we can help with? Testing, Merging, Coding something? Let's do it! |
Hate to beat a dead horse here, but... [HORSE] O=('-'Q) But seriously, what are we waiting for? |
@scottux if you are in a position to help direct message me on Gitter and we can discuss it. My work doesn't let me work on this, so I am limited to free time and I didn't have much over the last year due to the timing of illness and holidays, I have an operation at the end of this week and two to three weeks after that I should be in a way better position than now. |
@gazpachoking Do you mind fixing the merge conflicts here? |
@@ -9,6 +9,7 @@ angular.module('schemaForm').config(['schemaFormDecoratorsProvider', function(de | |||
console.log('fieldset children frag', children.childNodes) | |||
args.fieldFrag.childNode.appendChild(children); | |||
}},*/ | |||
formselect: {template: base + 'formselect.html', replace: false}, |
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 has now moved to https://github.com/json-schema-form/angular-schema-form-bootstrap
@@ -0,0 +1,10 @@ | |||
<div sf-form-select="form" class="schema-form-form-select {{form.htmlClass}}"> |
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 has now moved to https://github.com/json-schema-form/angular-schema-form-bootstrap
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.
We should probably create a simple one for core.
return type[1]; | ||
if (type[1] === 'null') | ||
return type[0]; | ||
if (Array.isArray(type) && type.length > 1) { |
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 has now moved to https://github.com/json-schema-form/json-schema-form-core
This PR adds a new basic form type 'formselect', which can show different forms based on a dropdown. I have it wired up to automatically use the formselect with a schema that has oneOf, anyOf, or types as an array. The extendSchemas helper is added (based on code from json editor) to combine schemas together.
It's very rough, as I just got it working and I'm not an html or js guy. Thought I'd get it out here for feedback, or if anyone wants to fork this and run with it, that sounds good to me too.
TODO: