Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
R
rjs_json_form
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Rafael Monnerat
rjs_json_form
Commits
08eb2298
Commit
08eb2298
authored
May 01, 2018
by
Boris Kocherov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
split gadget to one top and recursive children
parent
88b6a880
Changes
5
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
1435 additions
and
1374 deletions
+1435
-1374
Gruntfile.js
Gruntfile.js
+1
-0
gadget_json_generated_form.html
gadget_json_generated_form.html
+3
-2
gadget_json_generated_form.js
gadget_json_generated_form.js
+106
-1372
gadget_json_generated_form_child.html
gadget_json_generated_form_child.html
+20
-0
gadget_json_generated_form_child.js
gadget_json_generated_form_child.js
+1305
-0
No files found.
Gruntfile.js
View file @
08eb2298
...
...
@@ -65,6 +65,7 @@ module.exports = function (grunt) {
},
client
:
{
src
:
[
'
gadget_json_generated_form.js
'
,
'
gadget_json_generated_form_child.js
'
,
'
demo/gadget_demo_json_schema_form.js
'
],
directives
:
{
maxlen
:
79
,
...
...
gadget_json_generated_form.html
View file @
08eb2298
...
...
@@ -14,7 +14,8 @@
<script
src=
"gadget_json_generated_form.js"
type=
"text/javascript"
></script>
</head>
<body>
<div
data-json-path=
"/"
>
</div>
<div
data-gadget-url=
"gadget_json_generated_form_child.html"
data-gadget-scope=
"json_form_child"
data-gadget-sandbox=
"public"
></div>
</body>
</html>
gadget_json_generated_form.js
View file @
08eb2298
...
...
@@ -3,8 +3,7 @@
(
function
(
window
,
document
,
location
,
rJS
,
RSVP
,
jIO
,
tv4
)
{
"
use strict
"
;
var
render_object
,
expandSchema
;
var
expandSchema
;
function
decodeJsonPointer
(
_str
)
{
// https://tools.ietf.org/html/rfc6901#section-5
...
...
@@ -142,139 +141,86 @@
}
}
function
getDocumentType
(
doc
)
{
if
(
doc
instanceof
Array
)
{
return
"
array
"
;
}
return
typeof
doc
;
function
loadJSONSchema
(
g
,
url
,
path
)
{
var
protocol
,
download_url
,
hash
,
schema_url_map
;
// XXX need use `id` property
if
(
!
path
)
{
path
=
"
/
"
;
}
function
getDocumentSchema
(
doc
)
{
var
type
=
getDocumentType
(
doc
),
schema
=
{
type
:
type
url
=
convertUrlToAbsolute
(
g
,
path
,
url
,
window
.
location
);
download_url
=
url
.
origin
+
url
.
pathname
;
schema_url_map
=
{
"
http://json-schema.org/draft-04/schema
"
:
"
json-schema/schema4.json
"
,
"
http://json-schema.org/draft-06/schema
"
:
"
json-schema/schema6.json
"
,
"
http://json-schema.org/draft-07/schema
"
:
"
json-schema/schema7.json
"
,
"
http://json-schema.org/schema
"
:
"
json-schema/schema7.json
"
};
if
(
type
===
"
array
"
)
{
schema
.
maxItems
=
0
;
}
else
if
(
type
===
"
object
"
)
{
schema
.
additionalProperties
=
false
;
}
else
{
schema
.
readOnly
=
true
;
}
return
schema
;
}
function
render_selection
(
schema
,
json_document
)
{
var
input
=
document
.
createElement
(
"
select
"
),
option
,
i
,
enum_arr
=
schema
[
'
enum
'
];
input
.
size
=
1
;
if
(
schema
.
default
)
{
if
(
json_document
===
undefined
)
{
json_document
=
schema
.
default
;
}
}
else
{
option
=
document
.
createElement
(
"
option
"
);
option
.
value
=
""
;
if
(
json_document
===
undefined
)
{
option
.
selected
=
true
;
}
input
.
appendChild
(
option
);
}
for
(
i
=
0
;
i
<
enum_arr
.
length
;
i
+=
1
)
{
if
(
enum_arr
.
hasOwnProperty
(
i
))
{
option
=
document
.
createElement
(
"
option
"
);
option
.
value
=
enum_arr
[
i
];
option
.
textContent
=
enum_arr
[
i
];
if
(
enum_arr
[
i
]
===
json_document
)
{
option
.
selected
=
true
;
if
(
schema_url_map
.
hasOwnProperty
(
download_url
))
{
url
=
new
URL
(
schema_url_map
[
download_url
],
g
.
__path
);
download_url
=
url
.
origin
+
url
.
pathname
;
}
input
.
appendChild
(
option
);
protocol
=
url
.
protocol
;
if
(
protocol
===
"
http:
"
||
protocol
===
"
https:
"
)
{
if
(
window
.
location
.
protocol
!==
protocol
)
{
throw
new
Error
(
"
You cannot mixed http and https calls
"
);
}
}
return
input
;
hash
=
url
.
hash
;
url
=
url
.
href
;
return
RSVP
.
Queue
()
.
push
(
function
()
{
if
(
g
.
props
.
schema_cache
.
hasOwnProperty
(
download_url
))
{
return
g
.
props
.
schema_cache
[
download_url
];
}
function
render_boolean
(
schema
,
json_document
)
{
var
input
,
schema_for_selection
=
{
type
:
"
boolean
"
,
enum
:
[
true
,
false
]
return
downloadJSON
(
download_url
)
.
push
(
function
(
json
)
{
g
.
props
.
schema_cache
[
download_url
]
=
json
;
return
json
;
});
})
.
push
(
function
(
json
)
{
return
resolveLocalReference
(
json
,
hash
);
})
.
push
(
undefined
,
function
(
err
)
{
// XXX it will be great to have ability convert json_pointers(hash)
// in line numbers for pointed to line in rich editors.
// we can use https://github.com/vtrushin/json-to-ast for it
var
url_from_pointed
=
convertToRealWorldSchemaPath
(
g
,
path
),
schema_a
=
document
.
createElement
(
"
a
"
),
pointed_a
=
document
.
createElement
(
"
a
"
);
schema_a
.
setAttribute
(
"
href
"
,
download_url
);
schema_a
.
text
=
(
new
URL
(
download_url
)).
pathname
;
pointed_a
.
setAttribute
(
"
href
"
,
url_from_pointed
);
pointed_a
.
text
=
(
new
URL
(
url_from_pointed
)).
pathname
;
g
.
props
.
schema_resolve_errors
[
url_from_pointed
]
=
{
schemaPath
:
path
,
message
:
[
document
.
createTextNode
(
"
schema error:
"
),
document
.
createTextNode
(
err
.
message
),
schema_a
,
document
.
createTextNode
(
"
pointed from schema:
"
),
pointed_a
]
};
// XXX change json_document on open is not correct @bk
if
(
json_document
===
"
true
"
)
{
json_document
=
true
;
}
if
(
json_document
===
"
false
"
)
{
json_document
=
false
;
}
if
(
getDocumentType
(
schema
.
default
)
===
"
boolean
"
)
{
schema_for_selection
.
default
=
schema
.
default
;
}
input
=
render_selection
(
schema_for_selection
,
json_document
);
return
input
;
}
function
render_textarea
(
json_field
,
default_value
,
data_format
)
{
var
input
=
document
.
createElement
(
"
textarea
"
);
if
(
default_value
!==
undefined
)
{
if
(
default_value
instanceof
Array
)
{
input
.
value
=
default_value
.
join
(
"
\n
"
);
return
null
;
// schema part can't be null
})
.
push
(
function
(
schema_part
)
{
// console.log(path);
if
(
schema_part
===
null
)
{
// if resolving schema part contain errors
// use {} as failback
schema_part
=
{};
}
else
{
input
.
value
=
default_value
;
}
}
input
[
"
data-format
"
]
=
data_format
;
return
input
;
}
function
addSubForm
(
options
)
{
var
input_element
=
options
.
element
,
g
=
options
.
gadget
,
property_name
,
parent_path
,
scope
;
scope
=
"
j
"
+
Math
.
random
().
toString
(
36
).
substr
(
2
,
9
);
if
(
options
.
parent_type
!==
"
array
"
)
{
parent_path
=
options
.
path
;
property_name
=
options
.
property_name
;
if
(
!
property_name
)
{
property_name
=
input_element
.
value
;
}
if
(
!
property_name
)
{
// XXX notify user
// you can't create property without property_name
return
RSVP
.
Queue
();
}
if
(
g
.
props
.
objects
[
parent_path
].
hasOwnProperty
(
property_name
)
&&
g
.
props
.
objects
[
parent_path
][
property_name
]
!==
""
)
{
// XXX notify user
// you can't create property with existed property_name
return
RSVP
.
Queue
();
}
if
(
input_element
)
{
input_element
.
value
=
""
;
}
// save map url only for correctly resolved schema
// otherwise we have issue in convertToRealWorldSchemaPath
g
.
props
.
schema_map
[
path
]
=
url
;
}
return
g
.
declareGadget
(
'
gadget_json_generated_form.html
'
,
{
scope
:
scope
})
.
push
(
function
(
form_gadget
)
{
form_gadget
.
element
.
setAttribute
(
"
data-gadget-parent-scope
"
,
g
.
element
.
getAttribute
(
"
data-gadget-scope
"
));
if
(
options
.
parent_type
!==
"
array
"
)
{
g
.
props
.
objects
[
parent_path
][
property_name
]
=
scope
;
form_gadget
.
element
.
setAttribute
(
"
data-json-parent
"
,
parent_path
);
form_gadget
.
element
.
setAttribute
(
"
data-json-property-name
"
,
property_name
);
}
return
form_gadget
.
renderForm
({
type
:
options
.
type
,
schema
:
options
.
schema_part
,
schema_path
:
options
.
schema_path
,
document
:
options
.
default_dict
,
display_label
:
options
.
parent_type
!==
"
array
"
,
scope
:
scope
});
schemaPushSchemaPart
(
g
.
props
.
schema
,
path
,
schema_part
);
// console.log(g.props.schema[""]);
return
schema_part
;
});
}
...
...
@@ -314,1006 +260,105 @@
schema_path
:
arr
[
i
][
x
].
schema_path
};
}
else
{
if
(
schema
===
true
)
{
schema
=
{};
}
if
(
next_schema
===
true
)
{
next_schema
=
{};
}
// copy before change
schema
=
JSON
.
parse
(
JSON
.
stringify
(
schema
));
for
(
key
in
next_schema
)
{
if
(
next_schema
.
hasOwnProperty
(
key
))
{
if
(
schema
.
hasOwnProperty
(
key
))
{
// XXX need use many many rules for merging
schema
[
key
]
=
next_schema
[
key
];
}
else
{
schema
[
key
]
=
next_schema
[
key
];
}
}
}
schema_item
=
{
schema
:
schema
,
schema_path
:
arr
[
i
][
x
].
schema_path
};
}
summ_arr
.
push
(
schema_item
);
}
}
arr
[
i
+
1
]
=
summ_arr
;
}
return
arr
[
arr
.
length
-
1
];
});
}
function
anyOf
(
g
,
schema_array
,
schema_path
)
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
var
i
,
arr
=
[];
for
(
i
=
0
;
i
<
schema_array
.
length
;
i
+=
1
)
{
arr
.
push
(
expandSchema
(
g
,
schema_array
[
i
],
schema_path
+
'
/anyOf/
'
+
i
.
toString
()));
}
return
RSVP
.
all
(
arr
);
})
.
push
(
function
(
arr
)
{
var
i
,
z
,
schema_arr
=
[];
for
(
i
=
0
;
i
<
arr
.
length
;
i
+=
1
)
{
for
(
z
=
0
;
z
<
arr
[
i
].
length
;
z
+=
1
)
{
if
(
arr
[
i
][
z
].
schema
===
true
)
{
// or(any, restricted, restricted) simplify to any
return
[
arr
[
i
][
z
]];
}
schema_arr
.
push
(
arr
[
i
][
z
]);
}
}
return
schema_arr
;
});
}
expandSchema
=
function
(
g
,
schema
,
schema_path
,
ref
)
{
// XXX `if then else` construction can be simplify to
// anyOf(allOf(if_schema, then_schema), else_schema)
// and realized by existed rails
if
(
schema
===
undefined
)
{
schema
=
true
;
}
if
(
schema
.
anyOf
!==
undefined
)
{
return
anyOf
(
g
,
schema
.
anyOf
,
schema_path
);
}
if
(
schema
.
allOf
!==
undefined
)
{
return
allOf
(
g
,
schema
.
allOf
,
schema_path
);
}
if
(
schema
.
$ref
)
{
return
g
.
loadJSONSchema
(
schema
.
$ref
,
schema_path
)
.
push
(
function
(
schema_part
)
{
return
expandSchema
(
g
,
schema_part
,
schema_path
,
schema
.
$ref
);
});
}
return
RSVP
.
Queue
()
.
push
(
function
()
{
return
[{
title
:
ref
||
schema
.
title
,
schema
:
schema
,
schema_path
:
schema_path
}];
});
};
function
expandSchemaForField
(
g
,
schema
,
schema_path
,
for_required
)
{
return
g
.
markSchemaFieldAsRequired
(
schema_path
,
for_required
)
.
push
(
function
()
{
return
expandSchema
(
g
,
schema
,
schema_path
);
});
}
function
expandProperties
(
g
,
properties
,
schema_path
,
required
)
{
var
ret_obj
=
{};
return
RSVP
.
Queue
()
.
push
(
function
()
{
var
property_name
,
arr
=
[];
function
addPropertyName
(
p_name
)
{
return
function
(
schema_array
)
{
ret_obj
[
p_name
]
=
schema_array
;
};
}
for
(
property_name
in
properties
)
{
if
(
properties
.
hasOwnProperty
(
property_name
))
{
arr
.
push
(
expandSchemaForField
(
g
,
properties
[
property_name
],
schema_path
+
encodeJsonPointer
(
property_name
),
required
.
indexOf
(
property_name
)
>=
0
)
.
push
(
addPropertyName
(
property_name
))
);
}
}
return
RSVP
.
all
(
arr
);
})
.
push
(
function
()
{
return
ret_obj
;
});
}
function
checkSchemaArrOneChoise
(
schema_arr
)
{
if
(
schema_arr
.
length
===
1
)
{
if
(
schema_arr
[
0
].
type
instanceof
Array
)
{
return
schema_arr
[
0
].
type
.
length
<=
1
;
}
return
true
;
}
return
false
;
}
function
convertExpandedProperties2array
(
properties
)
{
var
property_name
,
arr
=
[],
i
,
schema_array
;
for
(
property_name
in
properties
)
{
if
(
properties
.
hasOwnProperty
(
property_name
))
{
schema_array
=
properties
[
property_name
];
for
(
i
=
0
;
i
<
schema_array
.
length
;
i
+=
1
)
{
// add propertyName to title
if
(
schema_array
[
i
].
title
&&
schema_array
.
length
>
1
)
{
schema_array
[
i
].
title
=
property_name
+
'
/
'
+
schema_array
[
i
].
title
;
}
else
{
schema_array
[
i
].
title
=
property_name
;
}
// add propertyName to schemaItem
schema_array
[
i
].
property_name
=
property_name
;
arr
.
push
(
schema_array
[
i
]);
}
}
}
return
arr
;
}
function
schemaArrFilteredByDocument
(
schema_arr
,
json_document
)
{
var
i
,
flag
,
ret_arr
=
[],
schema
;
if
(
schema_arr
.
length
===
1
)
{
return
schema_arr
[
0
];
}
if
(
json_document
!==
undefined
)
{
for
(
i
=
0
;
i
<
schema_arr
.
length
;
i
+=
1
)
{
schema
=
schema_arr
[
i
].
schema
;
if
(
schema
===
true
)
{
flag
=
true
;
}
else
if
(
schema
===
false
)
{
flag
=
false
;
}
else
{
flag
=
tv4
.
validate
(
json_document
,
schema
);
}
if
(
flag
)
{
ret_arr
.
push
(
schema_arr
[
i
]);
}
}
if
(
ret_arr
.
length
===
0
)
{
// XXX find schema more compatible with document
return
schema_arr
[
0
];
}
}
// XXX if (ret_arr.length > 1) notify user
return
ret_arr
[
0
];
}
function
checkValidityAndNotifyChange
(
g
)
{
return
RSVP
.
all
([
g
.
checkValidity
(),
g
.
notifyChange
()
]);
}
function
render_schema_selector
(
gadget
,
title
,
schema_arr
,
event
,
rerender
)
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
var
schema_alternatives
=
[],
schema_item
,
description
,
i
,
z
,
type
;
function
generateItemsForAny
(
property_name
,
schema_path
)
{
var
desc
,
types
=
[
"
string
"
,
"
number
"
,
"
boolean
"
,
"
array
"
,
"
object
"
,
"
null
"
],
ii
;
if
(
property_name
)
{
desc
=
property_name
+
"
#
"
;
}
else
{
desc
=
""
;
}
for
(
ii
=
0
;
ii
<
types
.
length
;
ii
+=
1
)
{
schema_alternatives
.
push
({
title
:
desc
+
types
[
ii
],
value
:
{
property_name
:
property_name
,
schema
:
{
type
:
types
[
ii
]
},
schema_path
:
schema_path
}
});
}
}
for
(
i
=
0
;
i
<
schema_arr
.
length
;
i
+=
1
)
{
schema_item
=
schema_arr
[
i
];
description
=
schema_item
.
title
;
if
(
schema_item
.
schema
===
true
)
{
generateItemsForAny
(
schema_item
.
property_name
,
schema_item
.
schema_path
);
}
else
if
(
getDocumentType
(
schema_item
.
schema
.
type
)
===
"
array
"
)
{
description
=
description
||
schema_item
.
schema
.
description
;
for
(
z
=
0
;
z
<
schema_item
.
schema
.
type
.
length
;
z
+=
1
)
{
type
=
schema_item
.
schema
.
type
[
z
];
schema_alternatives
.
push
({
title
:
description
+
'
#
'
+
type
,
value
:
{
type
:
type
,
property_name
:
schema_item
.
property_name
,
schema_path
:
schema_item
.
schema_path
,
schema
:
schema_item
.
schema
}
});
}
}
else
{
description
=
description
||
schema_item
.
schema
.
type
||
schema_item
.
schema
.
description
;
schema_alternatives
.
push
({
title
:
description
,
value
:
{
property_name
:
schema_item
.
property_name
,
schema_path
:
schema_item
.
schema_path
,
schema
:
schema_item
.
schema
}
});
}
}
return
schema_alternatives
;
})
.
push
(
function
(
schema_alternatives
)
{
var
scope
=
'
s
'
+
Math
.
random
().
toString
(
36
).
substr
(
2
,
9
);
if
(
schema_alternatives
.
length
>
1
)
{
return
gadget
.
declareGadget
(
"
gadget_html5_select.html
"
,
{
scope
:
scope
})
.
push
(
function
(
g
)
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
var
x
,
item_list
=
[[
title
,
title
]],
item
;
if
(
rerender
)
{
return
rerender
(
g
,
schema_alternatives
);
}
for
(
x
=
0
;
x
<
schema_alternatives
.
length
;
x
+=
1
)
{
item
=
schema_alternatives
[
x
];
item_list
.
push
([
item
.
title
,
x
]);
}
return
{
name
:
scope
,
editable
:
true
,
hidden
:
item_list
.
length
===
0
,
value
:
item_list
[
0
][
1
],
item_list
:
item_list
};
})
.
push
(
function
(
render_options
)
{
gadget
.
props
.
add_custom_data
[
scope
]
=
{
element
:
g
.
element
,
event
:
function
()
{
return
g
.
getContent
()
.
push
(
function
(
value
)
{
return
event
(
schema_alternatives
[
value
[
scope
]].
value
);
})
.
push
(
function
()
{
return
checkValidityAndNotifyChange
(
gadget
);
})
.
push
(
function
()
{
if
(
rerender
)
{
return
rerender
(
g
,
schema_alternatives
);
}
return
render_options
;
})
.
push
(
function
(
render_options
)
{
return
g
.
render
(
render_options
);
});
},
rerender
:
function
()
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
if
(
rerender
)
{
return
rerender
(
g
,
schema_alternatives
);
}
return
render_options
;
})
.
push
(
function
(
render_options
)
{
return
g
.
render
(
render_options
);
});
}
};
return
g
.
render
(
render_options
);
})
//not need if gadget_html5_select.render return element
.
push
(
function
()
{
return
g
.
element
;
});
});
}
if
(
schema_alternatives
.
length
===
1
)
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
if
(
rerender
)
{
return
rerender
(
undefined
,
schema_alternatives
);
}
return
true
;
})
.
push
(
function
(
ret
)
{
var
input
=
document
.
createElement
(
"
button
"
);
input
.
setAttribute
(
"
class
"
,
"
ui-shadow-inset ui-btn ui-btn-inline ui-corner-all
"
+
"
ui-btn-icon-notext ui-icon-btn ui-icon-plus ui-input-btn
"
);
input
.
type
=
"
button
"
;
input
.
hidden
=
!
ret
;
gadget
.
props
.
add_buttons
.
push
({
element
:
input
,
event
:
function
()
{
return
event
(
schema_alternatives
[
0
].
value
)
.
push
(
function
()
{
if
(
rerender
)
{
return
rerender
(
undefined
,
schema_alternatives
);
}
return
true
;
})
.
push
(
function
(
r
)
{
input
.
hidden
=
!
r
;
return
checkValidityAndNotifyChange
(
gadget
);
});
},
rerender
:
function
()
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
if
(
rerender
)
{
return
rerender
(
undefined
,
schema_alternatives
);
}
return
true
;
})
.
push
(
function
(
r
)
{
input
.
hidden
=
!
r
;
});
}
});
return
input
;
});
}
return
RSVP
.
Queue
()
.
push
(
function
()
{
return
document
.
createElement
(
"
div
"
);
});
});
}
function
render_array
(
gadget
,
schema
,
json_document
,
root
,
path
,
schema_path
)
{
var
div
,
div_input
,
input
,
minItems
=
schema
.
minItems
||
0
;
div
=
document
.
createElement
(
"
div
"
);
div
.
setAttribute
(
"
class
"
,
"
subfield
"
);
div
.
title
=
schema
.
description
;
div_input
=
document
.
createElement
(
"
div
"
);
div_input
.
setAttribute
(
"
class
"
,
"
input
"
);
function
element_append
(
child
)
{
if
(
child
)
{
input
.
parentNode
.
insertBefore
(
child
,
input
);
}
}
function
div_append
(
child
)
{
if
(
child
)
{
div_input
.
appendChild
(
child
);
}
}
// XXX add failback rendering if json_document not array
// input = render_textarea(schema, default_value, "array");
return
expandSchemaForField
(
gadget
,
schema
.
items
,
schema_path
+
'
/items
'
,
minItems
!==
0
)
.
push
(
function
(
schema_arr
)
{
var
queue
=
RSVP
.
Queue
(),
i
,
len
=
0
;
// XXX rewrite loading document for anyOf schema
if
(
json_document
)
{
for
(
i
=
0
;
i
<
json_document
.
length
;
i
=
i
+
1
)
{
queue
.
push
(
addSubForm
.
bind
(
gadget
,
{
gadget
:
gadget
,
parent_type
:
'
array
'
,
schema_path
:
schema_path
+
'
/items
'
,
schema_part
:
schema_arr
,
default_dict
:
json_document
[
i
]
})
)
.
push
(
div_append
);
}
len
=
json_document
.
length
;
}
if
(
checkSchemaArrOneChoise
(
schema_arr
)
&&
minItems
>
len
)
{
for
(
i
=
0
;
i
<
(
minItems
-
len
);
i
+=
1
)
{
queue
.
push
(
addSubForm
.
bind
(
gadget
,
{
gadget
:
gadget
,
parent_type
:
'
array
'
,
schema_path
:
schema_arr
[
0
].
schema_path
,
schema_part
:
schema_arr
[
0
].
schema
})
)
.
push
(
div_append
);
}
}
queue
.
push
(
render_schema_selector
.
bind
(
gadget
,
gadget
,
"
add item to array
"
,
schema_arr
,
function
(
value
)
{
return
addSubForm
({
gadget
:
gadget
,
parent_type
:
'
array
'
,
type
:
value
.
type
,
schema_path
:
value
.
schema_path
,
schema_part
:
value
.
schema
})
.
push
(
element_append
);
}));
return
queue
;
})
.
push
(
function
(
element
)
{
// var maxItems = schema.maxItems;
input
=
element
;
// XXX update on every add/delete item
// input.hidden = maxItems !== undefined && json_document.length >= maxItems;
div_input
.
appendChild
(
input
);
div
.
appendChild
(
div_input
);
root
.
appendChild
(
div
);
});
}
function
render_field
(
gadget
,
key
,
path
,
json_field
,
default_value
,
root
,
schema_path
,
options
)
{
var
type
,
div
,
label
,
div_input
,
span_info
,
error_message
,
input
,
first_path
,
queue
=
RSVP
.
Queue
();
if
(
json_field
instanceof
Array
)
{
json_field
=
schemaArrFilteredByDocument
(
json_field
,
default_value
);
schema_path
=
json_field
.
schema_path
;
json_field
=
json_field
.
schema
;
}
options
=
options
||
{};
type
=
options
.
type
;
if
(
path
&&
key
)
{
first_path
=
path
+
encodeJsonPointer
(
key
);
}
else
{
first_path
=
""
;
}
if
(
json_field
===
undefined
)
{
json_field
=
getDocumentSchema
(
default_value
);
}
if
(
getDocumentType
(
json_field
.
type
)
===
"
string
"
)
{
type
=
json_field
.
type
;
}
// else json_field.type is array so we use type
if
(
type
===
undefined
&&
default_value
!==
undefined
)
{
type
=
getDocumentType
(
default_value
);
}
// XXX bad peace of code
// i do not sure that type can be computed so
// but our schema in slapos bad
if
(
!
type
)
{
if
(
json_field
.
properties
&&
json_field
.
required
&&
json_field
.
required
.
length
>
0
)
{
type
=
"
object
"
;
}
}
div
=
document
.
createElement
(
"
div
"
);
div
.
setAttribute
(
"
class
"
,
"
subfield
"
);
div
.
title
=
json_field
.
description
;
// if (key && !first_path) {
if
(
false
)
{
// XXX;
label
=
document
.
createElement
(
"
input
"
);
label
.
value
=
key
;
gadget
.
props
.
property_name_edit
=
label
;
}
else
{
label
=
document
.
createElement
(
"
label
"
);
label
.
textContent
=
[
key
,
json_field
.
title
]
.
filter
(
function
(
v
)
{
return
v
;
})
.
join
(
"
"
);
}
div
.
appendChild
(
label
);
div_input
=
document
.
createElement
(
"
div
"
);
div_input
.
setAttribute
(
"
id
"
,
gadget
.
element
.
getAttribute
(
"
data-gadget-scope
"
)
+
first_path
+
'
/
'
);
div_input
.
setAttribute
(
"
class
"
,
"
input
"
);
if
(
json_field
.
enum
!==
undefined
)
{
input
=
render_selection
(
json_field
,
default_value
);
}
if
(
type
===
"
boolean
"
)
{
input
=
render_boolean
(
json_field
,
default_value
);
}
if
(
!
input
&&
[
"
string
"
,
"
integer
"
,
"
number
"
].
indexOf
(
type
)
>=
0
)
{
if
(
json_field
.
contentMediaType
===
"
text/plain
"
)
{
input
=
render_textarea
(
json_field
,
default_value
,
"
string
"
);
}
else
{
input
=
document
.
createElement
(
"
input
"
);
if
(
default_value
!==
undefined
)
{
input
.
value
=
default_value
;
}
if
(
type
===
"
integer
"
||
type
===
"
number
"
)
{
if
(
default_value
===
undefined
&&
typeof
json_field
.
default
===
"
number
"
)
{
input
.
value
=
json_field
.
default
;
}
input
.
type
=
"
number
"
;
}
else
{
if
(
default_value
===
undefined
&&
typeof
json_field
.
default
===
"
string
"
)
{
input
.
value
=
json_field
.
default
;
}
input
.
type
=
"
text
"
;
if
(
json_field
.
pattern
)
{
input
.
pattern
=
json_field
.
pattern
;
}
if
(
json_field
.
format
===
'
uri
'
)
{
input
.
type
=
"
url
"
;
input
.
spellcheck
=
false
;
}
}
}
}
if
(
type
===
"
array
"
)
{
queue
=
render_array
(
gadget
,
json_field
,
default_value
,
div_input
,
first_path
+
'
/
'
,
schema_path
);
gadget
.
props
.
arrays
[
first_path
+
'
/
'
]
=
div
;
}
if
(
type
===
"
object
"
)
{
queue
.
push
(
function
()
{
return
render_object
(
gadget
,
json_field
,
default_value
,
div_input
,
first_path
+
'
/
'
,
schema_path
);
});
}
if
(
input
)
{
// object and array excluded from
// gadget.props.inputs not contain values
gadget
.
props
.
inputs
.
push
(
input
);
input
.
name
=
first_path
;
input
.
required
=
options
.
required
;
// XXX for gui
//input.setAttribute("class", "slapos-parameter");
div_input
.
appendChild
(
input
);
}
else
{
div
.
setAttribute
(
"
data-json-path
"
,
first_path
+
'
/
'
);
div
.
setAttribute
(
"
data-json-type
"
,
type
);
}
if
(
json_field
.
info
!==
undefined
)
{
span_info
=
document
.
createElement
(
"
span
"
);
span_info
.
textContent
=
json_field
.
info
;
div_input
.
appendChild
(
span_info
);
}
error_message
=
document
.
createElement
(
"
span
"
);
error_message
.
setAttribute
(
"
class
"
,
"
error
"
);
error_message
.
hidden
=
true
;
div_input
.
appendChild
(
error_message
);
div
.
appendChild
(
div_input
);
return
queue
.
push
(
function
()
{
root
.
appendChild
(
div
);
return
div
;
});
}
function
render_object_additionalProperty
(
g
,
title
,
json_document
,
path
,
schema
,
schema_path
,
used
,
element_append
)
{
var
div
,
div_input
,
input
;
div
=
document
.
createElement
(
"
div
"
);
div
.
setAttribute
(
"
class
"
,
"
subfield
"
);
// div.title = json_field.description;
div_input
=
document
.
createElement
(
"
div
"
);
div_input
.
setAttribute
(
"
class
"
,
"
input
"
);
input
=
document
.
createElement
(
"
input
"
);
input
.
type
=
"
text
"
;
input
.
placeholder
=
"
name of
"
+
title
;
div_input
.
appendChild
(
input
);
return
expandSchemaForField
(
g
,
schema
,
schema_path
)
.
push
(
function
(
schema_arr
)
{
var
queue
=
RSVP
.
Queue
(),
property_name
;
for
(
property_name
in
json_document
)
{
if
(
json_document
.
hasOwnProperty
(
property_name
)
&&
!
used
.
hasOwnProperty
(
property_name
))
{
used
[
property_name
]
=
""
;
queue
.
push
(
addSubForm
.
bind
(
g
,
{
gadget
:
g
,
property_name
:
property_name
,
path
:
path
,
schema_path
:
schema_path
,
schema_part
:
schema_arr
,
default_dict
:
json_document
[
property_name
]
})
)
.
push
(
element_append
);
}
}
queue
.
push
(
function
()
{
return
render_schema_selector
(
g
,
"
add
"
+
title
,
schema_arr
,
function
(
value
)
{
return
addSubForm
({
gadget
:
g
,
element
:
input
,
path
:
path
,
type
:
value
.
type
,
schema_path
:
value
.
schema_path
,
schema_part
:
value
.
schema
})
.
push
(
element_append
);
});
});
return
queue
;
})
.
push
(
function
(
input
)
{
div_input
.
appendChild
(
input
);
div
.
appendChild
(
div_input
);
return
div
;
});
}
render_object
=
function
(
g
,
json_field
,
default_dict
,
root
,
path
,
schema_path
)
{
var
required
=
json_field
.
required
||
[],
used_properties
=
{},
properties
,
selector
=
{};
g
.
props
.
objects
[
path
]
=
used_properties
;
function
element_append
(
child
)
{
if
(
child
)
{
// insert additionalProperty before selector
selector
.
element
.
parentNode
.
insertBefore
(
child
,
selector
.
element
);
}
}
function
root_append
(
child
)
{
root
.
appendChild
(
child
);
}
if
(
default_dict
===
undefined
)
{
default_dict
=
{};
}
return
expandProperties
(
g
,
json_field
.
properties
,
schema_path
+
'
/properties/
'
,
required
)
.
push
(
function
(
ret
)
{
var
schema_arr
,
q
=
RSVP
.
Queue
(),
s_o
,
key
;
properties
=
ret
;
for
(
key
in
properties
)
{
if
(
properties
.
hasOwnProperty
(
key
))
{
schema_arr
=
properties
[
key
];
s_o
=
schemaArrFilteredByDocument
(
schema_arr
,
default_dict
[
key
]);
if
(
required
.
indexOf
(
key
)
>=
0
&&
schema_arr
.
length
===
1
)
{
used_properties
[
key
]
=
false
;
q
.
push
(
render_field
.
bind
(
g
,
g
,
key
,
path
,
s_o
.
schema
,
default_dict
[
key
],
root
,
s_o
.
schema_path
,
{
required
:
true
})
);
}
if
(
default_dict
.
hasOwnProperty
(
key
)
&&
!
used_properties
.
hasOwnProperty
(
key
))
{
used_properties
[
key
]
=
""
;
q
.
push
(
addSubForm
.
bind
(
g
,
{
gadget
:
g
,
property_name
:
key
,
path
:
path
,
schema_path
:
s_o
.
schema_path
,
schema_part
:
s_o
.
schema
,
default_dict
:
default_dict
[
key
]
})
)
.
push
(
root_append
);
}
}
}
return
q
;
})
.
push
(
function
()
{
var
schema_arr
=
convertExpandedProperties2array
(
properties
);
return
render_schema_selector
(
g
,
"
add property
"
,
schema_arr
,
function
(
value
)
{
used_properties
[
value
.
property_name
]
=
""
;
return
addSubForm
({
gadget
:
g
,
property_name
:
value
.
property_name
,
path
:
path
,
type
:
value
.
type
,
schema_path
:
value
.
schema_path
,
schema_part
:
value
.
schema
})
.
push
(
function
(
element
)
{
var
s_e
=
selector
.
element
;
if
(
s_e
)
{
s_e
.
parentNode
.
insertBefore
(
element
,
s_e
);
}
});
},
function
(
gadget_s
,
schema_alternatives
)
{
var
x
,
item_list
=
[[
"
add property
"
,
"
add property
"
]],
item
;
if
(
schema_alternatives
)
{
for
(
x
=
0
;
x
<
schema_alternatives
.
length
;
x
+=
1
)
{
item
=
schema_alternatives
[
x
];
if
(
!
used_properties
.
hasOwnProperty
(
item
.
value
.
property_name
))
{
item_list
.
push
([
item
.
title
,
x
]);
}
}
if
(
gadget_s
)
{
return
{
name
:
gadget_s
.
element
.
getAttribute
(
'
data-gadget-scope
'
),
editable
:
true
,
hidden
:
item_list
.
length
===
1
,
value
:
item_list
[
0
][
1
],
item_list
:
item_list
};
}
return
item_list
.
length
>
1
;
}
});
})
.
push
(
function
(
element
)
{
selector
.
element
=
element
;
return
root_append
(
element
);
})
.
push
(
function
()
{
var
queue
=
RSVP
.
Queue
(),
additionalProperties
;
if
(
json_field
.
patternProperties
!==
undefined
)
{
// XXX need loop on any pattern properties
if
(
json_field
.
patternProperties
[
'
.*
'
]
!==
undefined
)
{
queue
.
push
(
render_object_additionalProperty
.
bind
(
g
,
g
,
"
.* property
"
,
default_dict
,
path
,
json_field
.
patternProperties
[
'
.*
'
],
schema_path
+
'
/patternProperties/.*
'
,
used_properties
,
element_append
))
.
push
(
root_append
);
if
(
schema
===
true
)
{
schema
=
{};
}
if
(
next_schema
===
true
)
{
next_schema
=
{};
}
if
(
json_field
.
additionalProperties
===
undefined
)
{
additionalProperties
=
true
;
// copy before change
schema
=
JSON
.
parse
(
JSON
.
stringify
(
schema
));
for
(
key
in
next_schema
)
{
if
(
next_schema
.
hasOwnProperty
(
key
))
{
if
(
schema
.
hasOwnProperty
(
key
))
{
// XXX need use many many rules for merging
schema
[
key
]
=
next_schema
[
key
];
}
else
{
additionalProperties
=
json_field
.
additionalProperties
;
}
if
(
additionalProperties
!==
false
)
{
queue
.
push
(
render_object_additionalProperty
.
bind
(
g
,
g
,
"
additional property
"
,
default_dict
,
path
,
additionalProperties
,
schema_path
+
'
/additionalProperties
'
,
used_properties
,
element_append
))
.
push
(
root_append
);
}
return
queue
;
})
.
push
(
function
()
{
var
key
,
queue
=
RSVP
.
Queue
();
for
(
key
in
default_dict
)
{
if
(
default_dict
.
hasOwnProperty
(
key
))
{
if
(
!
used_properties
.
hasOwnProperty
(
key
))
{
queue
.
push
(
addSubForm
.
bind
(
g
,
{
gadget
:
g
,
property_name
:
key
,
path
:
path
,
schema_path
:
""
,
schema_part
:
undefined
,
default_dict
:
default_dict
[
key
]
})
)
.
push
(
root_append
);
schema
[
key
]
=
next_schema
[
key
];
}
}
}
return
queue
;
});
schema_item
=
{
schema
:
schema
,
schema_path
:
arr
[
i
][
x
].
schema_path
};
function
getFormValuesAsJSONDict
(
g
)
{
var
multi_level_dict
=
{
""
:
{}},
scope
,
options
=
g
.
props
,
array
,
path
,
key
,
i
,
len
,
queue
=
RSVP
.
Queue
();
function
convertOnMultiLevel
(
d
,
key
,
value
)
{
var
ii
,
kk
,
key_list
=
key
.
split
(
"
/
"
);
for
(
ii
=
0
;
ii
<
key_list
.
length
;
ii
+=
1
)
{
kk
=
decodeJsonPointer
(
key_list
[
ii
]);
if
(
ii
===
key_list
.
length
-
1
)
{
if
(
value
!==
undefined
)
{
d
[
kk
]
=
value
;
}
else
{
return
d
[
kk
];
}
}
else
{
if
(
!
d
.
hasOwnProperty
(
kk
))
{
d
[
kk
]
=
{};
}
d
=
d
[
kk
]
;
summ_arr
.
push
(
schema_item
)
;
}
}
arr
[
i
+
1
]
=
summ_arr
;
}
function
recursiveGetContent
(
scope
,
path
)
{
queue
.
push
(
function
()
{
return
g
.
getDeclaredGadget
(
scope
);
})
.
push
(
function
(
gadget
)
{
return
gadget
.
getContent
();
})
.
push
(
function
(
jdict
)
{
convertOnMultiLevel
(
multi_level_dict
,
path
,
jdict
);
return
arr
[
arr
.
length
-
1
];
});
}
function
getContentAndPushArray
(
scope
,
parent
_path
)
{
queue
function
anyOf
(
g
,
schema_array
,
schema
_path
)
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
return
g
.
getDeclaredGadget
(
scope
);
})
.
push
(
function
(
gadget
)
{
return
gadget
.
getContent
();
})
.
push
(
function
(
jdict
)
{
var
arr
=
convertOnMultiLevel
(
multi_level_dict
,
parent_path
);
if
(
!
(
arr
instanceof
Array
))
{
var
i
,
arr
=
[];
convertOnMultiLevel
(
multi_level_dict
,
parent_path
,
arr
);
}
arr
.
push
(
jdict
);
});
}
for
(
path
in
options
.
arrays
)
{
if
(
options
.
arrays
.
hasOwnProperty
(
path
))
{
array
=
options
.
arrays
[
path
]
.
querySelectorAll
(
"
div[data-gadget-parent-scope='
"
+
g
.
element
.
getAttribute
(
"
data-gadget-scope
"
)
+
"
']
"
);
len
=
array
.
length
;
for
(
i
=
0
;
i
<
len
;
i
=
i
+
1
)
{
getContentAndPushArray
(
array
[
i
].
getAttribute
(
'
data-gadget-scope
'
),
// slice remove concluding '/'
path
.
slice
(
0
,
-
1
)
);
}
}
}
for
(
path
in
options
.
objects
)
{
if
(
options
.
objects
.
hasOwnProperty
(
path
))
{
for
(
key
in
options
.
objects
[
path
])
{
if
(
options
.
objects
[
path
].
hasOwnProperty
(
key
))
{
scope
=
options
.
objects
[
path
][
key
];
if
(
scope
)
{
recursiveGetContent
(
scope
,
path
+
encodeJsonPointer
(
key
));
for
(
i
=
0
;
i
<
schema_array
.
length
;
i
+=
1
)
{
arr
.
push
(
expandSchema
(
g
,
schema_array
[
i
],
schema_path
+
'
/anyOf/
'
+
i
.
toString
()));
}
return
RSVP
.
all
(
arr
);
})
.
push
(
function
(
arr
)
{
var
i
,
z
,
schema_arr
=
[];
for
(
i
=
0
;
i
<
arr
.
length
;
i
+=
1
)
{
for
(
z
=
0
;
z
<
arr
[
i
].
length
;
z
+=
1
)
{
if
(
arr
[
i
][
z
].
schema
===
true
)
{
// or(any, restricted, restricted) simplify to any
return
[
arr
[
i
][
z
]];
}
schema_arr
.
push
(
arr
[
i
][
z
]);
}
}
return
schema_arr
;
});
}
return
queue
.
push
(
function
()
{
var
json_dict
=
{},
k
;
g
.
props
.
inputs
.
forEach
(
function
(
input
)
{
if
(
input
.
name
===
""
||
input
.
value
!==
""
)
{
if
(
input
.
type
===
'
number
'
)
{
json_dict
[
input
.
name
]
=
parseInt
(
input
.
value
,
10
);
}
else
if
(
input
.
value
===
"
true
"
)
{
json_dict
[
input
.
name
]
=
true
;
}
else
if
(
input
.
value
===
"
false
"
)
{
json_dict
[
input
.
name
]
=
false
;
}
else
if
(
input
.
tagName
===
"
TEXTAREA
"
)
{
if
(
input
[
"
data-format
"
]
===
"
string
"
)
{
json_dict
[
input
.
name
]
=
input
.
value
;
}
else
{
json_dict
[
input
.
name
]
=
input
.
value
.
split
(
'
\n
'
);
expandSchema
=
function
(
g
,
schema
,
schema_path
,
ref
)
{
// XXX `if then else` construction can be simplify to
// anyOf(allOf(if_schema, then_schema), else_schema)
// and realized by existed rails
if
(
schema
===
undefined
)
{
schema
=
true
;
}
}
else
{
json_dict
[
input
.
name
]
=
input
.
value
;
if
(
schema
.
anyOf
!==
undefined
)
{
return
anyOf
(
g
,
schema
.
anyOf
,
schema_path
)
;
}
if
(
schema
.
allOf
!==
undefined
)
{
return
allOf
(
g
,
schema
.
allOf
,
schema_path
);
}
if
(
schema
.
$ref
)
{
return
loadJSONSchema
(
g
,
schema
.
$ref
,
schema_path
)
.
push
(
function
(
schema_part
)
{
return
expandSchema
(
g
,
schema_part
,
schema_path
,
schema
.
$ref
);
});
for
(
k
in
json_dict
)
{
if
(
json_dict
.
hasOwnProperty
(
k
))
{
convertOnMultiLevel
(
multi_level_dict
,
k
,
json_dict
[
k
]);
}
}
return
multi_level_dict
[
""
];
return
RSVP
.
Queue
()
.
push
(
function
()
{
return
[{
title
:
ref
||
schema
.
title
,
schema
:
schema
,
schema_path
:
schema_path
}];
});
}
}
;
function
getSubGadgetElement
(
g
,
scope
)
{
return
g
.
element
.
querySelector
(
"
div[data-gadget-scope='
"
+
scope
+
"
']
"
);
function
expandSchemaForField
(
g
,
schema
,
schema_path
,
for_required
)
{
var
required_stack
,
prev_field_path
;
if
(
for_required
)
{
prev_field_path
=
getMaxPathInDict
(
g
.
props
.
schema_required_urls
,
schema_path
);
required_stack
=
g
.
props
.
schema_required_urls
[
prev_field_path
];
}
else
{
required_stack
=
[];
}
g
.
props
.
schema_required_urls
[
schema_path
]
=
required_stack
;
return
expandSchema
(
g
,
schema
,
schema_path
);
}
rJS
(
window
)
...
...
@@ -1323,166 +368,29 @@
g
.
options
=
{};
})
.
declareAcquiredMethod
(
"
notifyChange
"
,
"
notifyChange
"
)
.
declareAcquiredMethod
(
"
renameChildrenParent
"
,
"
renameChildren
"
)
.
allowPublicAcquisition
(
"
renameChildren
"
,
function
(
opt_arr
,
scope
)
{
var
property_name
,
objects
=
this
.
props
.
objects
,
new_name
=
opt_arr
[
0
],
element
=
getSubGadgetElement
(
this
,
scope
),
parent
=
element
.
getAttribute
(
'
data-json-parent
'
);
if
(
objects
.
hasOwnProperty
(
parent
))
{
parent
=
objects
[
parent
];
if
(
parent
.
hasOwnProperty
(
new_name
))
{
throw
new
Error
(
"
property already exist
"
);
}
// XXX validate property if property pattern
for
(
property_name
in
parent
)
{
if
(
parent
.
hasOwnProperty
(
property_name
)
&&
parent
[
property_name
]
===
scope
)
{
delete
parent
[
property_name
];
parent
[
new_name
]
=
scope
;
return
new_name
;
}
}
throw
new
Error
(
"
gadget not found for renaming
"
);
}
})
.
declareMethod
(
"
rename
"
,
function
(
new_name
,
event
)
{
var
g
=
this
,
name
=
g
.
element
.
getAttribute
(
'
data-json-property-name
'
);
return
this
.
renameChildrenParent
(
new_name
)
.
push
(
function
()
{
return
g
.
element
.
setAttribute
(
'
data-json-property-name
'
,
new_name
);
})
.
push
(
undefined
,
function
(
error
)
{
// XXX notify user
event
.
srcElement
.
value
=
name
;
event
.
srcElement
.
focus
();
});
})
.
declareAcquiredMethod
(
"
selfRemove
"
,
"
deleteChildren
"
)
.
allowPublicAcquisition
(
"
deleteChildren
"
,
function
(
arr
,
scope
)
{
var
g
=
this
,
key
,
i
,
button_list
=
this
.
props
.
add_buttons
,
objects
=
this
.
props
.
objects
,
element
=
getSubGadgetElement
(
g
,
scope
),
parent
=
element
.
getAttribute
(
"
data-json-parent
"
),
tasks
=
[];
if
(
objects
.
hasOwnProperty
(
parent
))
{
parent
=
objects
[
parent
];
for
(
key
in
parent
)
{
if
(
parent
.
hasOwnProperty
(
key
)
&&
parent
[
key
]
===
scope
)
{
delete
parent
[
key
];
}
}
}
element
.
parentNode
.
removeChild
(
element
);
for
(
key
in
g
.
props
.
add_custom_data
)
{
if
(
g
.
props
.
add_custom_data
.
hasOwnProperty
(
key
))
{
tasks
.
push
(
g
.
props
.
add_custom_data
[
key
].
rerender
());
}
}
for
(
i
=
0
;
i
<
button_list
.
length
;
i
=
i
+
1
)
{
tasks
.
push
(
button_list
[
i
].
rerender
());
}
tasks
.
push
(
checkValidityAndNotifyChange
(
g
));
return
RSVP
.
Queue
()
.
push
(
function
()
{
return
RSVP
.
all
(
tasks
);
});
})
.
declareMethod
(
'
getElementByPath
'
,
function
(
data_path
)
{
var
g
=
this
,
array
,
path
,
scope
,
key
,
slash_count
=
0
,
slash_count_next
,
bingo
,
idx
,
options
=
g
.
props
;
if
(
data_path
!==
"
/
"
)
{
for
(
path
in
options
.
arrays
)
{
if
(
options
.
arrays
.
hasOwnProperty
(
path
)
&&
data_path
.
startsWith
(
path
))
{
array
=
options
.
arrays
[
path
]
.
querySelectorAll
(
"
div[data-gadget-parent-scope='
"
+
g
.
element
.
getAttribute
(
"
data-gadget-scope
"
)
+
"
']
"
);
data_path
=
data_path
.
slice
(
path
.
length
).
split
(
"
/
"
);
idx
=
data_path
[
0
];
data_path
=
"
/
"
+
data_path
.
slice
(
1
).
join
(
"
/
"
);
bingo
=
array
[
idx
].
getAttribute
(
'
data-gadget-scope
'
);
}
}
if
(
bingo
)
{
return
g
.
getDeclaredGadget
(
bingo
)
.
push
(
function
(
gadget
)
{
return
gadget
.
getElementByPath
(
data_path
);
});
}
for
(
path
in
options
.
objects
)
{
if
(
options
.
objects
.
hasOwnProperty
(
path
)
&&
data_path
.
startsWith
(
path
))
{
slash_count_next
=
path
.
split
(
"
/
"
).
length
-
1
;
if
(
slash_count_next
>
slash_count
)
{
bingo
=
path
;
slash_count
=
slash_count_next
;
}
}
}
if
(
bingo
)
{
path
=
options
.
objects
[
bingo
];
for
(
key
in
path
)
{
if
(
path
.
hasOwnProperty
(
key
))
{
if
(
data_path
.
startsWith
(
bingo
+
encodeJsonPointer
(
key
)))
{
data_path
=
data_path
.
slice
(
bingo
.
length
+
encodeJsonPointer
(
key
).
length
);
if
(
!
data_path
)
{
data_path
=
"
/
"
;
}
scope
=
path
[
key
];
}
}
}
}
if
(
scope
)
{
return
g
.
getDeclaredGadget
(
scope
)
.
push
(
function
(
gadget
)
{
return
gadget
.
getElementByPath
(
data_path
);
});
}
}
return
RSVP
.
Queue
()
.
push
(
function
()
{
return
document
.
getElementById
(
g
.
element
.
getAttribute
(
"
data-gadget-scope
"
)
+
data_path
);
});
.
allowPublicAcquisition
(
"
notifyChange
"
,
function
()
{
return
this
.
notifyChange
();
})
.
declareAcquiredMethod
(
"
notifyValid
"
,
"
notifyValid
"
)
.
declareAcquiredMethod
(
"
notifyInvalid
"
,
"
notifyInvalid
"
)
.
declareAcquiredMethod
(
"
checkValidityParent
"
,
"
checkValidityTop
"
)
.
allowPublicAcquisition
(
"
checkValidityTop
"
,
function
(
arr
)
{
.
allowPublicAcquisition
(
"
checkValidity
"
,
function
(
arr
)
{
return
this
.
checkValidity
(
arr
[
0
]);
})
.
declareMethod
(
'
checkValidity
'
,
function
(
json_document
)
{
// XXX need use local schema and local json document
// in every subgadget to take into account user anyOf choice
// and so more precisely point to issue
var
g
=
this
;
if
(
!
g
.
props
.
toplevel
)
{
return
g
.
checkValidityParent
(
json_document
);
}
// return RSVP.Queue();
var
g
=
this
.
props
.
form_gadget
,
gadget
=
this
;
return
RSVP
.
Queue
()
.
push
(
function
()
{
if
(
json_document
===
undefined
)
{
return
g
etFormValuesAsJSONDict
(
g
);
return
g
.
getContent
(
);
}
return
json_document
;
})
.
push
(
function
(
json_d
)
{
return
tv4
.
validateMultiple
(
json_d
,
g
.
props
.
schema
[
""
]);
return
tv4
.
validateMultiple
(
json_d
,
g
adget
.
props
.
schema
[
""
]);
})
.
push
(
function
(
validation
)
{
var
i
,
...
...
@@ -1491,7 +399,7 @@
span
,
tasks
=
[],
errors
=
[],
schema_resolve_errors
=
g
.
props
.
schema_resolve_errors
,
schema_resolve_errors
=
g
adget
.
props
.
schema_resolve_errors
,
errors_block
=
g
.
element
.
querySelector
(
"
div.error-block
"
);
if
(
errors_block
)
{
...
...
@@ -1517,7 +425,7 @@
errors
=
errors
.
concat
(
validation
.
missing
);
if
(
errors
.
length
===
0
)
{
return
g
.
notifyValid
()
return
g
adget
.
notifyValid
()
.
push
(
function
()
{
return
false
;
});
...
...
@@ -1576,176 +484,13 @@
.
push
(
function
()
{
g
.
element
.
insertBefore
(
errors_block
,
g
.
element
.
firstChild
);
})
.
push
(
g
.
notifyInvalid
.
bind
(
g
))
.
push
(
g
adget
.
notifyInvalid
.
bind
(
gadget
))
.
push
(
function
()
{
return
false
;
});
});
})
.
allowPublicAcquisition
(
"
notifyValid
"
,
function
(
arr
,
sub_scope
)
{
return
true
;
})
.
allowPublicAcquisition
(
"
notifyChange
"
,
function
(
arr
,
sub_scope
)
{
var
g
=
this
,
opt
=
arr
[
0
],
event_object
;
event_object
=
g
.
props
.
add_custom_data
[
sub_scope
];
if
(
event_object
&&
opt
.
type
===
"
change
"
)
{
return
event_object
.
event
();
}
return
g
.
notifyChange
();
})
.
declareMethod
(
'
renderForm
'
,
function
(
options
)
{
var
g
=
this
,
property_name
=
g
.
element
.
getAttribute
(
'
data-json-property-name
'
),
schema
=
options
.
schema
,
delete_button
,
root
;
g
.
props
.
inputs
=
[];
g
.
props
.
add_buttons
=
[];
g
.
props
.
add_custom_data
=
{};
g
.
props
.
arrays
=
{};
g
.
props
.
objects
=
{};
g
.
props
.
path
=
options
.
path
;
// self gadget scope
if
(
!
property_name
||
!
options
.
display_label
)
{
property_name
=
""
;
}
root
=
g
.
element
.
querySelector
(
'
[data-json-path="/"]
'
);
if
(
!
root
)
{
root
=
g
.
element
;
}
while
(
root
.
firstChild
)
{
root
.
removeChild
(
root
.
firstChild
);
}
if
(
!
g
.
props
.
toplevel
)
{
delete_button
=
document
.
createElement
(
"
button
"
);
delete_button
.
setAttribute
(
"
class
"
,
"
ui-shadow-inset ui-btn ui-btn-inline ui-corner-all
"
+
"
ui-btn-icon-notext ui-icon-btn ui-icon-trash-o ui-input-btn
"
);
delete_button
.
type
=
"
button
"
;
delete_button
.
name
=
options
.
path
;
g
.
props
.
delete_button
=
delete_button
;
root
.
appendChild
(
delete_button
);
}
return
render_field
(
g
,
property_name
,
""
,
schema
,
options
.
document
,
root
,
options
.
schema_path
,
{
type
:
options
.
type
})
.
push
(
function
()
{
g
.
listenEvents
();
return
g
.
element
;
});
})
.
declareMethod
(
"
loadJSONSchema
"
,
function
(
url
,
path
)
{
var
g
=
this
,
protocol
,
download_url
,
hash
,
schema_url_map
;
if
(
!
g
.
props
.
toplevel
)
{
return
g
.
loadJSONSchemaParent
(
url
,
path
);
}
// XXX need use $id
if
(
!
path
)
{
path
=
"
/
"
;
}
url
=
convertUrlToAbsolute
(
g
,
path
,
url
,
window
.
location
);
download_url
=
url
.
origin
+
url
.
pathname
;
schema_url_map
=
{
"
http://json-schema.org/draft-04/schema
"
:
"
json-schema/schema4.json
"
,
"
http://json-schema.org/draft-06/schema
"
:
"
json-schema/schema6.json
"
,
"
http://json-schema.org/draft-07/schema
"
:
"
json-schema/schema7.json
"
,
"
http://json-schema.org/schema
"
:
"
json-schema/schema7.json
"
};
if
(
schema_url_map
.
hasOwnProperty
(
download_url
))
{
url
=
new
URL
(
schema_url_map
[
download_url
],
g
.
__path
);
download_url
=
url
.
origin
+
url
.
pathname
;
}
protocol
=
url
.
protocol
;
if
(
protocol
===
"
http:
"
||
protocol
===
"
https:
"
)
{
if
(
window
.
location
.
protocol
!==
protocol
)
{
throw
new
Error
(
"
You cannot mixed http and https calls
"
);
}
}
hash
=
url
.
hash
;
url
=
url
.
href
;
return
RSVP
.
Queue
()
.
push
(
function
()
{
if
(
g
.
props
.
schema_cache
.
hasOwnProperty
(
download_url
))
{
return
g
.
props
.
schema_cache
[
download_url
];
}
return
downloadJSON
(
download_url
)
.
push
(
function
(
json
)
{
g
.
props
.
schema_cache
[
download_url
]
=
json
;
return
json
;
});
})
.
push
(
function
(
json
)
{
return
resolveLocalReference
(
json
,
hash
);
})
.
push
(
undefined
,
function
(
err
)
{
// XXX it will be great to have ability convert json_pointers(hash)
// in line numbers for pointed to line in rich editors.
// we can use https://github.com/vtrushin/json-to-ast for it
var
url_from_pointed
=
convertToRealWorldSchemaPath
(
g
,
path
),
schema_a
=
document
.
createElement
(
"
a
"
),
pointed_a
=
document
.
createElement
(
"
a
"
);
schema_a
.
setAttribute
(
"
href
"
,
download_url
);
schema_a
.
text
=
(
new
URL
(
download_url
)).
pathname
;
pointed_a
.
setAttribute
(
"
href
"
,
url_from_pointed
);
pointed_a
.
text
=
(
new
URL
(
url_from_pointed
)).
pathname
;
g
.
props
.
schema_resolve_errors
[
url_from_pointed
]
=
{
schemaPath
:
path
,
message
:
[
document
.
createTextNode
(
"
schema error:
"
),
document
.
createTextNode
(
err
.
message
),
schema_a
,
document
.
createTextNode
(
"
pointed from schema:
"
),
pointed_a
]
};
return
null
;
// schema part can't be null
})
.
push
(
function
(
schema_part
)
{
// console.log(path);
if
(
schema_part
===
null
)
{
// if resolving schema part contain errors
// use {} as failback
schema_part
=
{};
}
else
{
// save map url only for correctly resolved schema
// otherwise we have issue in convertToRealWorldSchemaPath
g
.
props
.
schema_map
[
path
]
=
url
;
}
schemaPushSchemaPart
(
g
.
props
.
schema
,
path
,
schema_part
);
// console.log(g.props.schema[""]);
return
schema_part
;
});
})
.
declareAcquiredMethod
(
"
loadJSONSchemaParent
"
,
"
loadJSONSchemaTop
"
)
.
allowPublicAcquisition
(
"
loadJSONSchemaTop
"
,
function
(
arr
)
{
return
this
.
loadJSONSchema
(
arr
[
0
],
arr
[
1
]);
})
.
declareMethod
(
"
markSchemaFieldAsRequired
"
,
function
(
schema_path
,
for_required
)
{
var
g
=
this
,
required_stack
,
prev_field_path
;
if
(
!
g
.
props
.
toplevel
)
{
return
g
.
markSchemaFieldAsRequiredParent
(
schema_path
,
for_required
);
}
// previous downloaded path
if
(
for_required
)
{
prev_field_path
=
getMaxPathInDict
(
g
.
props
.
schema_required_urls
,
schema_path
);
required_stack
=
g
.
props
.
schema_required_urls
[
prev_field_path
];
}
else
{
required_stack
=
[];
}
g
.
props
.
schema_required_urls
[
schema_path
]
=
required_stack
;
})
.
declareAcquiredMethod
(
"
markSchemaFieldAsRequiredParent
"
,
"
markSchemaFieldAsRequiredTop
"
)
.
allowPublicAcquisition
(
"
markSchemaFieldAsRequiredTop
"
,
function
(
arr
)
{
return
this
.
markSchemaFieldAsRequired
(
arr
[
0
],
arr
[
1
]);
})
.
declareMethod
(
'
render
'
,
function
(
options
)
{
var
g
=
this
;
g
.
props
.
toplevel
=
true
;
...
...
@@ -1766,24 +511,26 @@
// message: error_message can be array containing dom elements
// }
g
.
props
.
schema_resolve_errors
=
{};
return
RSVP
.
Queue
()
.
push
(
function
()
{
return
g
.
getDeclaredGadget
(
"
json_form_child
"
)
.
push
(
function
(
json_form_child
)
{
g
.
props
.
form_gadget
=
json_form_child
;
if
(
options
.
schema
)
{
return
options
.
schema
;
}
var
schema_url
=
options
.
schema_url
||
(
options
.
value
&&
options
.
value
.
$schema
);
if
(
schema_url
)
{
return
g
.
loadJSONSchema
(
schema_url
);
return
loadJSONSchema
(
g
,
schema_url
);
}
return
{};
})
.
push
(
function
(
schema
)
{
g
.
options
.
schema
=
schema
;
return
g
.
renderForm
({
return
g
.
props
.
form_gadget
.
renderForm
({
schema
:
schema
,
schema_path
:
""
,
document
:
options
.
value
document
:
options
.
value
,
top
:
true
});
})
.
push
(
function
()
{
...
...
@@ -1793,7 +540,9 @@
return
g
;
});
})
.
allowPublicAcquisition
(
"
expandSchema
"
,
function
(
arr
)
{
return
expandSchemaForField
(
this
,
arr
[
0
],
arr
[
1
],
arr
[
2
]);
})
.
onEvent
(
'
click
'
,
function
(
evt
)
{
if
(
evt
.
target
===
this
.
props
.
delete_button
)
{
return
this
.
selfRemove
(
evt
);
...
...
@@ -1813,28 +562,13 @@
}
}
})
.
onEvent
(
'
change
'
,
function
(
evt
)
{
if
(
evt
.
target
===
this
.
props
.
property_name_edit
)
{
return
this
.
rename
(
this
.
props
.
property_name_edit
.
value
,
evt
);
}
var
field_list
=
this
.
props
.
inputs
,
i
;
for
(
i
=
0
;
i
<
field_list
.
length
;
i
=
i
+
1
)
{
if
(
evt
.
target
===
field_list
[
i
])
{
return
checkValidityAndNotifyChange
(
this
);
}
}
})
.
declareJob
(
'
listenEvents
'
,
function
()
{
// XXX Disable
return
;
})
.
declareMethod
(
'
getContent
'
,
function
()
{
var
g
=
this
;
return
getFormValuesAsJSONDict
(
g
);
return
this
.
props
.
form_gadget
.
getContent
();
});
}(
window
,
document
,
location
,
rJS
,
RSVP
,
jIO
,
tv4
));
\ No newline at end of file
gadget_json_generated_form_child.html
0 → 100644
View file @
08eb2298
<!DOCTYPE html>
<html>
<head>
<meta
charset=
"utf-8"
>
<meta
name=
"viewport"
content=
"width=device-width, initial-scale=1"
>
<title>
ERP5
</title>
<link
rel=
"shortcut icon"
href=
"favicon.ico"
>
<link
rel=
"stylesheet"
href=
"gadget_erp5_nojqm.css"
>
<link
href=
"gadget_erp5_page_slap_parameter_form.css"
rel=
"stylesheet"
type=
"text/css"
/>
<script
src=
"rsvp.js"
type=
"text/javascript"
></script>
<script
src=
"tv4.js"
type=
"text/javascript"
></script>
<script
src=
"renderjs.js"
type=
"text/javascript"
></script>
<script
src=
"jio.js"
type=
"text/javascript"
></script>
<script
src=
"gadget_json_generated_form_child.js"
type=
"text/javascript"
></script>
</head>
<body>
<div
data-json-path=
"/"
>
</div>
</body>
</html>
gadget_json_generated_form_child.js
0 → 100644
View file @
08eb2298
/*jslint nomen: true, maxlen: 200, indent: 2, maxerr: 100*/
/*global window, document, URL, rJS, RSVP, jIO, tv4, location */
(
function
(
window
,
document
,
location
,
rJS
,
RSVP
,
jIO
,
tv4
)
{
"
use strict
"
;
var
render_object
;
function
decodeJsonPointer
(
_str
)
{
// https://tools.ietf.org/html/rfc6901#section-5
return
_str
.
replace
(
/~1/g
,
'
/
'
).
replace
(
/~0/g
,
'
~
'
);
}
function
encodeJsonPointer
(
_str
)
{
// https://tools.ietf.org/html/rfc6901#section-5
return
_str
.
replace
(
/~/g
,
'
~0
'
).
replace
(
/
\/
/g
,
'
~1
'
);
}
function
getDocumentType
(
doc
)
{
if
(
doc
instanceof
Array
)
{
return
"
array
"
;
}
return
typeof
doc
;
}
function
getDocumentSchema
(
doc
)
{
var
type
=
getDocumentType
(
doc
),
schema
=
{
type
:
type
};
if
(
type
===
"
array
"
)
{
schema
.
maxItems
=
0
;
}
else
if
(
type
===
"
object
"
)
{
schema
.
additionalProperties
=
false
;
}
else
{
schema
.
readOnly
=
true
;
}
return
schema
;
}
function
render_selection
(
schema
,
json_document
)
{
var
input
=
document
.
createElement
(
"
select
"
),
option
,
i
,
enum_arr
=
schema
[
'
enum
'
];
input
.
size
=
1
;
if
(
schema
.
default
)
{
if
(
json_document
===
undefined
)
{
json_document
=
schema
.
default
;
}
}
else
{
option
=
document
.
createElement
(
"
option
"
);
option
.
value
=
""
;
if
(
json_document
===
undefined
)
{
option
.
selected
=
true
;
}
input
.
appendChild
(
option
);
}
for
(
i
=
0
;
i
<
enum_arr
.
length
;
i
+=
1
)
{
if
(
enum_arr
.
hasOwnProperty
(
i
))
{
option
=
document
.
createElement
(
"
option
"
);
option
.
value
=
enum_arr
[
i
];
option
.
textContent
=
enum_arr
[
i
];
if
(
enum_arr
[
i
]
===
json_document
)
{
option
.
selected
=
true
;
}
input
.
appendChild
(
option
);
}
}
return
input
;
}
function
render_boolean
(
schema
,
json_document
)
{
var
input
,
schema_for_selection
=
{
type
:
"
boolean
"
,
enum
:
[
true
,
false
]
};
// XXX change json_document on open is not correct @bk
if
(
json_document
===
"
true
"
)
{
json_document
=
true
;
}
if
(
json_document
===
"
false
"
)
{
json_document
=
false
;
}
if
(
getDocumentType
(
schema
.
default
)
===
"
boolean
"
)
{
schema_for_selection
.
default
=
schema
.
default
;
}
input
=
render_selection
(
schema_for_selection
,
json_document
);
return
input
;
}
function
render_textarea
(
json_field
,
default_value
,
data_format
)
{
var
input
=
document
.
createElement
(
"
textarea
"
);
if
(
default_value
!==
undefined
)
{
if
(
default_value
instanceof
Array
)
{
input
.
value
=
default_value
.
join
(
"
\n
"
);
}
else
{
input
.
value
=
default_value
;
}
}
input
[
"
data-format
"
]
=
data_format
;
return
input
;
}
function
addSubForm
(
options
)
{
var
input_element
=
options
.
element
,
g
=
options
.
gadget
,
property_name
,
parent_path
,
scope
;
scope
=
"
j
"
+
Math
.
random
().
toString
(
36
).
substr
(
2
,
9
);
if
(
options
.
parent_type
!==
"
array
"
)
{
parent_path
=
options
.
path
;
property_name
=
options
.
property_name
;
if
(
!
property_name
)
{
property_name
=
input_element
.
value
;
}
if
(
!
property_name
)
{
// XXX notify user
// you can't create property without property_name
return
RSVP
.
Queue
();
}
if
(
g
.
props
.
objects
[
parent_path
].
hasOwnProperty
(
property_name
)
&&
g
.
props
.
objects
[
parent_path
][
property_name
]
!==
""
)
{
// XXX notify user
// you can't create property with existed property_name
return
RSVP
.
Queue
();
}
if
(
input_element
)
{
input_element
.
value
=
""
;
}
}
return
g
.
declareGadget
(
'
gadget_json_generated_form_child.html
'
,
{
scope
:
scope
})
.
push
(
function
(
form_gadget
)
{
form_gadget
.
element
.
setAttribute
(
"
data-gadget-parent-scope
"
,
g
.
element
.
getAttribute
(
"
data-gadget-scope
"
));
if
(
options
.
parent_type
!==
"
array
"
)
{
g
.
props
.
objects
[
parent_path
][
property_name
]
=
scope
;
form_gadget
.
element
.
setAttribute
(
"
data-json-parent
"
,
parent_path
);
form_gadget
.
element
.
setAttribute
(
"
data-json-property-name
"
,
property_name
);
}
return
form_gadget
.
renderForm
({
type
:
options
.
type
,
schema
:
options
.
schema_part
,
schema_path
:
options
.
schema_path
,
document
:
options
.
default_dict
,
display_label
:
options
.
parent_type
!==
"
array
"
,
scope
:
scope
});
});
}
function
expandProperties
(
g
,
properties
,
schema_path
,
required
)
{
var
ret_obj
=
{};
return
RSVP
.
Queue
()
.
push
(
function
()
{
var
property_name
,
arr
=
[];
function
addPropertyName
(
p_name
)
{
return
function
(
schema_array
)
{
ret_obj
[
p_name
]
=
schema_array
;
};
}
for
(
property_name
in
properties
)
{
if
(
properties
.
hasOwnProperty
(
property_name
))
{
arr
.
push
(
g
.
expandSchema
(
properties
[
property_name
],
schema_path
+
encodeJsonPointer
(
property_name
),
required
.
indexOf
(
property_name
)
>=
0
)
.
push
(
addPropertyName
(
property_name
))
);
}
}
return
RSVP
.
all
(
arr
);
})
.
push
(
function
()
{
return
ret_obj
;
});
}
function
checkSchemaArrOneChoise
(
schema_arr
)
{
if
(
schema_arr
.
length
===
1
)
{
if
(
schema_arr
[
0
].
type
instanceof
Array
)
{
return
schema_arr
[
0
].
type
.
length
<=
1
;
}
return
true
;
}
return
false
;
}
function
convertExpandedProperties2array
(
properties
)
{
var
property_name
,
arr
=
[],
i
,
schema_array
;
for
(
property_name
in
properties
)
{
if
(
properties
.
hasOwnProperty
(
property_name
))
{
schema_array
=
properties
[
property_name
];
for
(
i
=
0
;
i
<
schema_array
.
length
;
i
+=
1
)
{
// add propertyName to title
if
(
schema_array
[
i
].
title
&&
schema_array
.
length
>
1
)
{
schema_array
[
i
].
title
=
property_name
+
'
/
'
+
schema_array
[
i
].
title
;
}
else
{
schema_array
[
i
].
title
=
property_name
;
}
// add propertyName to schemaItem
schema_array
[
i
].
property_name
=
property_name
;
arr
.
push
(
schema_array
[
i
]);
}
}
}
return
arr
;
}
function
schemaArrFilteredByDocument
(
schema_arr
,
json_document
)
{
var
i
,
flag
,
ret_arr
=
[],
schema
;
if
(
schema_arr
.
length
===
1
)
{
return
schema_arr
[
0
];
}
if
(
json_document
!==
undefined
)
{
for
(
i
=
0
;
i
<
schema_arr
.
length
;
i
+=
1
)
{
schema
=
schema_arr
[
i
].
schema
;
if
(
schema
===
true
)
{
flag
=
true
;
}
else
if
(
schema
===
false
)
{
flag
=
false
;
}
else
{
flag
=
tv4
.
validate
(
json_document
,
schema
);
}
if
(
flag
)
{
ret_arr
.
push
(
schema_arr
[
i
]);
}
}
if
(
ret_arr
.
length
===
0
)
{
// XXX find schema more compatible with document
return
schema_arr
[
0
];
}
}
// XXX if (ret_arr.length > 1) notify user
return
ret_arr
[
0
];
}
function
checkValidityAndNotifyChange
(
g
)
{
return
RSVP
.
all
([
g
.
checkValidity
(),
g
.
notifyChange
()
]);
}
function
render_schema_selector
(
gadget
,
title
,
schema_arr
,
event
,
rerender
)
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
var
schema_alternatives
=
[],
schema_item
,
description
,
i
,
z
,
type
;
function
generateItemsForAny
(
property_name
,
schema_path
)
{
var
desc
,
types
=
[
"
string
"
,
"
number
"
,
"
boolean
"
,
"
array
"
,
"
object
"
,
"
null
"
],
ii
;
if
(
property_name
)
{
desc
=
property_name
+
"
#
"
;
}
else
{
desc
=
""
;
}
for
(
ii
=
0
;
ii
<
types
.
length
;
ii
+=
1
)
{
schema_alternatives
.
push
({
title
:
desc
+
types
[
ii
],
value
:
{
property_name
:
property_name
,
schema
:
{
type
:
types
[
ii
]
},
schema_path
:
schema_path
}
});
}
}
for
(
i
=
0
;
i
<
schema_arr
.
length
;
i
+=
1
)
{
schema_item
=
schema_arr
[
i
];
description
=
schema_item
.
title
;
if
(
schema_item
.
schema
===
true
)
{
generateItemsForAny
(
schema_item
.
property_name
,
schema_item
.
schema_path
);
}
else
if
(
getDocumentType
(
schema_item
.
schema
.
type
)
===
"
array
"
)
{
description
=
description
||
schema_item
.
schema
.
description
;
for
(
z
=
0
;
z
<
schema_item
.
schema
.
type
.
length
;
z
+=
1
)
{
type
=
schema_item
.
schema
.
type
[
z
];
schema_alternatives
.
push
({
title
:
description
+
'
#
'
+
type
,
value
:
{
type
:
type
,
property_name
:
schema_item
.
property_name
,
schema_path
:
schema_item
.
schema_path
,
schema
:
schema_item
.
schema
}
});
}
}
else
{
description
=
description
||
schema_item
.
schema
.
type
||
schema_item
.
schema
.
description
;
schema_alternatives
.
push
({
title
:
description
,
value
:
{
property_name
:
schema_item
.
property_name
,
schema_path
:
schema_item
.
schema_path
,
schema
:
schema_item
.
schema
}
});
}
}
return
schema_alternatives
;
})
.
push
(
function
(
schema_alternatives
)
{
var
scope
=
'
s
'
+
Math
.
random
().
toString
(
36
).
substr
(
2
,
9
);
if
(
schema_alternatives
.
length
>
1
)
{
return
gadget
.
declareGadget
(
"
gadget_html5_select.html
"
,
{
scope
:
scope
})
.
push
(
function
(
g
)
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
var
x
,
item_list
=
[[
title
,
title
]],
item
;
if
(
rerender
)
{
return
rerender
(
g
,
schema_alternatives
);
}
for
(
x
=
0
;
x
<
schema_alternatives
.
length
;
x
+=
1
)
{
item
=
schema_alternatives
[
x
];
item_list
.
push
([
item
.
title
,
x
]);
}
return
{
name
:
scope
,
editable
:
true
,
hidden
:
item_list
.
length
===
0
,
value
:
item_list
[
0
][
1
],
item_list
:
item_list
};
})
.
push
(
function
(
render_options
)
{
gadget
.
props
.
add_custom_data
[
scope
]
=
{
element
:
g
.
element
,
event
:
function
()
{
return
g
.
getContent
()
.
push
(
function
(
value
)
{
return
event
(
schema_alternatives
[
value
[
scope
]].
value
);
})
.
push
(
function
()
{
return
checkValidityAndNotifyChange
(
gadget
);
})
.
push
(
function
()
{
if
(
rerender
)
{
return
rerender
(
g
,
schema_alternatives
);
}
return
render_options
;
})
.
push
(
function
(
render_options
)
{
return
g
.
render
(
render_options
);
});
},
rerender
:
function
()
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
if
(
rerender
)
{
return
rerender
(
g
,
schema_alternatives
);
}
return
render_options
;
})
.
push
(
function
(
render_options
)
{
return
g
.
render
(
render_options
);
});
}
};
return
g
.
render
(
render_options
);
})
//not need if gadget_html5_select.render return element
.
push
(
function
()
{
return
g
.
element
;
});
});
}
if
(
schema_alternatives
.
length
===
1
)
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
if
(
rerender
)
{
return
rerender
(
undefined
,
schema_alternatives
);
}
return
true
;
})
.
push
(
function
(
ret
)
{
var
input
=
document
.
createElement
(
"
button
"
);
input
.
setAttribute
(
"
class
"
,
"
ui-shadow-inset ui-btn ui-btn-inline ui-corner-all
"
+
"
ui-btn-icon-notext ui-icon-btn ui-icon-plus ui-input-btn
"
);
input
.
type
=
"
button
"
;
input
.
hidden
=
!
ret
;
gadget
.
props
.
add_buttons
.
push
({
element
:
input
,
event
:
function
()
{
return
event
(
schema_alternatives
[
0
].
value
)
.
push
(
function
()
{
if
(
rerender
)
{
return
rerender
(
undefined
,
schema_alternatives
);
}
return
true
;
})
.
push
(
function
(
r
)
{
input
.
hidden
=
!
r
;
return
checkValidityAndNotifyChange
(
gadget
);
});
},
rerender
:
function
()
{
return
RSVP
.
Queue
()
.
push
(
function
()
{
if
(
rerender
)
{
return
rerender
(
undefined
,
schema_alternatives
);
}
return
true
;
})
.
push
(
function
(
r
)
{
input
.
hidden
=
!
r
;
});
}
});
return
input
;
});
}
return
RSVP
.
Queue
()
.
push
(
function
()
{
return
document
.
createElement
(
"
div
"
);
});
});
}
function
render_array
(
gadget
,
schema
,
json_document
,
root
,
path
,
schema_path
)
{
var
div
,
div_input
,
input
,
minItems
=
schema
.
minItems
||
0
;
div
=
document
.
createElement
(
"
div
"
);
div
.
setAttribute
(
"
class
"
,
"
subfield
"
);
div
.
title
=
schema
.
description
;
div_input
=
document
.
createElement
(
"
div
"
);
div_input
.
setAttribute
(
"
class
"
,
"
input
"
);
function
element_append
(
child
)
{
if
(
child
)
{
input
.
parentNode
.
insertBefore
(
child
,
input
);
}
}
function
div_append
(
child
)
{
if
(
child
)
{
div_input
.
appendChild
(
child
);
}
}
// XXX add failback rendering if json_document not array
// input = render_textarea(schema, default_value, "array");
return
gadget
.
expandSchema
(
schema
.
items
,
schema_path
+
'
/items
'
,
minItems
!==
0
)
.
push
(
function
(
schema_arr
)
{
var
queue
=
RSVP
.
Queue
(),
i
,
len
=
0
;
// XXX rewrite loading document for anyOf schema
if
(
json_document
)
{
for
(
i
=
0
;
i
<
json_document
.
length
;
i
=
i
+
1
)
{
queue
.
push
(
addSubForm
.
bind
(
gadget
,
{
gadget
:
gadget
,
parent_type
:
'
array
'
,
schema_path
:
schema_path
+
'
/items
'
,
schema_part
:
schema_arr
,
default_dict
:
json_document
[
i
]
})
)
.
push
(
div_append
);
}
len
=
json_document
.
length
;
}
if
(
checkSchemaArrOneChoise
(
schema_arr
)
&&
minItems
>
len
)
{
for
(
i
=
0
;
i
<
(
minItems
-
len
);
i
+=
1
)
{
queue
.
push
(
addSubForm
.
bind
(
gadget
,
{
gadget
:
gadget
,
parent_type
:
'
array
'
,
schema_path
:
schema_arr
[
0
].
schema_path
,
schema_part
:
schema_arr
[
0
].
schema
})
)
.
push
(
div_append
);
}
}
queue
.
push
(
render_schema_selector
.
bind
(
gadget
,
gadget
,
"
add item to array
"
,
schema_arr
,
function
(
value
)
{
return
addSubForm
({
gadget
:
gadget
,
parent_type
:
'
array
'
,
type
:
value
.
type
,
schema_path
:
value
.
schema_path
,
schema_part
:
value
.
schema
})
.
push
(
element_append
);
}));
return
queue
;
})
.
push
(
function
(
element
)
{
// var maxItems = schema.maxItems;
input
=
element
;
// XXX update on every add/delete item
// input.hidden = maxItems !== undefined && json_document.length >= maxItems;
div_input
.
appendChild
(
input
);
div
.
appendChild
(
div_input
);
root
.
appendChild
(
div
);
});
}
function
render_field
(
gadget
,
key
,
path
,
json_field
,
default_value
,
root
,
schema_path
,
options
)
{
var
type
,
div
,
label
,
div_input
,
span_info
,
error_message
,
input
,
first_path
,
queue
=
RSVP
.
Queue
();
if
(
json_field
instanceof
Array
)
{
json_field
=
schemaArrFilteredByDocument
(
json_field
,
default_value
);
schema_path
=
json_field
.
schema_path
;
json_field
=
json_field
.
schema
;
}
options
=
options
||
{};
type
=
options
.
type
;
if
(
path
&&
key
)
{
first_path
=
path
+
encodeJsonPointer
(
key
);
}
else
{
first_path
=
""
;
}
if
(
json_field
===
undefined
)
{
json_field
=
getDocumentSchema
(
default_value
);
}
if
(
getDocumentType
(
json_field
.
type
)
===
"
string
"
)
{
type
=
json_field
.
type
;
}
// else json_field.type is array so we use type
if
(
type
===
undefined
&&
default_value
!==
undefined
)
{
type
=
getDocumentType
(
default_value
);
}
// XXX bad peace of code
// i do not sure that type can be computed so
// but our schema in slapos bad
if
(
!
type
)
{
if
(
json_field
.
properties
&&
json_field
.
required
&&
json_field
.
required
.
length
>
0
)
{
type
=
"
object
"
;
}
}
div
=
document
.
createElement
(
"
div
"
);
div
.
setAttribute
(
"
class
"
,
"
subfield
"
);
div
.
title
=
json_field
.
description
;
// if (key && !first_path) {
if
(
false
)
{
// XXX;
label
=
document
.
createElement
(
"
input
"
);
label
.
value
=
key
;
gadget
.
props
.
property_name_edit
=
label
;
}
else
{
label
=
document
.
createElement
(
"
label
"
);
label
.
textContent
=
[
key
,
json_field
.
title
]
.
filter
(
function
(
v
)
{
return
v
;
})
.
join
(
"
"
);
}
div
.
appendChild
(
label
);
div_input
=
document
.
createElement
(
"
div
"
);
div_input
.
setAttribute
(
"
id
"
,
gadget
.
element
.
getAttribute
(
"
data-gadget-scope
"
)
+
first_path
+
'
/
'
);
div_input
.
setAttribute
(
"
class
"
,
"
input
"
);
if
(
json_field
.
enum
!==
undefined
)
{
input
=
render_selection
(
json_field
,
default_value
);
}
if
(
type
===
"
boolean
"
)
{
input
=
render_boolean
(
json_field
,
default_value
);
}
if
(
!
input
&&
[
"
string
"
,
"
integer
"
,
"
number
"
].
indexOf
(
type
)
>=
0
)
{
if
(
json_field
.
contentMediaType
===
"
text/plain
"
)
{
input
=
render_textarea
(
json_field
,
default_value
,
"
string
"
);
}
else
{
input
=
document
.
createElement
(
"
input
"
);
if
(
default_value
!==
undefined
)
{
input
.
value
=
default_value
;
}
if
(
type
===
"
integer
"
||
type
===
"
number
"
)
{
if
(
default_value
===
undefined
&&
typeof
json_field
.
default
===
"
number
"
)
{
input
.
value
=
json_field
.
default
;
}
input
.
type
=
"
number
"
;
}
else
{
if
(
default_value
===
undefined
&&
typeof
json_field
.
default
===
"
string
"
)
{
input
.
value
=
json_field
.
default
;
}
input
.
type
=
"
text
"
;
if
(
json_field
.
pattern
)
{
input
.
pattern
=
json_field
.
pattern
;
}
if
(
json_field
.
format
===
'
uri
'
)
{
input
.
type
=
"
url
"
;
input
.
spellcheck
=
false
;
}
}
}
}
if
(
type
===
"
array
"
)
{
queue
=
render_array
(
gadget
,
json_field
,
default_value
,
div_input
,
first_path
+
'
/
'
,
schema_path
);
gadget
.
props
.
arrays
[
first_path
+
'
/
'
]
=
div
;
}
if
(
type
===
"
object
"
)
{
queue
.
push
(
function
()
{
return
render_object
(
gadget
,
json_field
,
default_value
,
div_input
,
first_path
+
'
/
'
,
schema_path
);
});
}
if
(
input
)
{
// object and array excluded from
// gadget.props.inputs not contain values
gadget
.
props
.
inputs
.
push
(
input
);
input
.
name
=
first_path
;
input
.
required
=
options
.
required
;
// XXX for gui
//input.setAttribute("class", "slapos-parameter");
div_input
.
appendChild
(
input
);
}
else
{
div
.
setAttribute
(
"
data-json-path
"
,
first_path
+
'
/
'
);
div
.
setAttribute
(
"
data-json-type
"
,
type
);
}
if
(
json_field
.
info
!==
undefined
)
{
span_info
=
document
.
createElement
(
"
span
"
);
span_info
.
textContent
=
json_field
.
info
;
div_input
.
appendChild
(
span_info
);
}
error_message
=
document
.
createElement
(
"
span
"
);
error_message
.
setAttribute
(
"
class
"
,
"
error
"
);
error_message
.
hidden
=
true
;
div_input
.
appendChild
(
error_message
);
div
.
appendChild
(
div_input
);
return
queue
.
push
(
function
()
{
root
.
appendChild
(
div
);
return
div
;
});
}
function
render_object_additionalProperty
(
g
,
title
,
json_document
,
path
,
schema
,
schema_path
,
used
,
element_append
)
{
var
div
,
div_input
,
input
;
div
=
document
.
createElement
(
"
div
"
);
div
.
setAttribute
(
"
class
"
,
"
subfield
"
);
// div.title = json_field.description;
div_input
=
document
.
createElement
(
"
div
"
);
div_input
.
setAttribute
(
"
class
"
,
"
input
"
);
input
=
document
.
createElement
(
"
input
"
);
input
.
type
=
"
text
"
;
input
.
placeholder
=
"
name of
"
+
title
;
div_input
.
appendChild
(
input
);
return
g
.
expandSchema
(
schema
,
schema_path
)
.
push
(
function
(
schema_arr
)
{
var
queue
=
RSVP
.
Queue
(),
property_name
;
for
(
property_name
in
json_document
)
{
if
(
json_document
.
hasOwnProperty
(
property_name
)
&&
!
used
.
hasOwnProperty
(
property_name
))
{
used
[
property_name
]
=
""
;
queue
.
push
(
addSubForm
.
bind
(
g
,
{
gadget
:
g
,
property_name
:
property_name
,
path
:
path
,
schema_path
:
schema_path
,
schema_part
:
schema_arr
,
default_dict
:
json_document
[
property_name
]
})
)
.
push
(
element_append
);
}
}
queue
.
push
(
function
()
{
return
render_schema_selector
(
g
,
"
add
"
+
title
,
schema_arr
,
function
(
value
)
{
return
addSubForm
({
gadget
:
g
,
element
:
input
,
path
:
path
,
type
:
value
.
type
,
schema_path
:
value
.
schema_path
,
schema_part
:
value
.
schema
})
.
push
(
element_append
);
});
});
return
queue
;
})
.
push
(
function
(
input
)
{
div_input
.
appendChild
(
input
);
div
.
appendChild
(
div_input
);
return
div
;
});
}
render_object
=
function
(
g
,
json_field
,
default_dict
,
root
,
path
,
schema_path
)
{
var
required
=
json_field
.
required
||
[],
used_properties
=
{},
properties
,
selector
=
{};
g
.
props
.
objects
[
path
]
=
used_properties
;
function
element_append
(
child
)
{
if
(
child
)
{
// insert additionalProperty before selector
selector
.
element
.
parentNode
.
insertBefore
(
child
,
selector
.
element
);
}
}
function
root_append
(
child
)
{
root
.
appendChild
(
child
);
}
if
(
default_dict
===
undefined
)
{
default_dict
=
{};
}
return
expandProperties
(
g
,
json_field
.
properties
,
schema_path
+
'
/properties/
'
,
required
)
.
push
(
function
(
ret
)
{
var
schema_arr
,
q
=
RSVP
.
Queue
(),
s_o
,
key
;
properties
=
ret
;
for
(
key
in
properties
)
{
if
(
properties
.
hasOwnProperty
(
key
))
{
schema_arr
=
properties
[
key
];
s_o
=
schemaArrFilteredByDocument
(
schema_arr
,
default_dict
[
key
]);
if
(
required
.
indexOf
(
key
)
>=
0
&&
schema_arr
.
length
===
1
)
{
used_properties
[
key
]
=
false
;
q
.
push
(
render_field
.
bind
(
g
,
g
,
key
,
path
,
s_o
.
schema
,
default_dict
[
key
],
root
,
s_o
.
schema_path
,
{
required
:
true
})
);
}
if
(
default_dict
.
hasOwnProperty
(
key
)
&&
!
used_properties
.
hasOwnProperty
(
key
))
{
used_properties
[
key
]
=
""
;
q
.
push
(
addSubForm
.
bind
(
g
,
{
gadget
:
g
,
property_name
:
key
,
path
:
path
,
schema_path
:
s_o
.
schema_path
,
schema_part
:
s_o
.
schema
,
default_dict
:
default_dict
[
key
]
})
)
.
push
(
root_append
);
}
}
}
return
q
;
})
.
push
(
function
()
{
var
schema_arr
=
convertExpandedProperties2array
(
properties
);
return
render_schema_selector
(
g
,
"
add property
"
,
schema_arr
,
function
(
value
)
{
used_properties
[
value
.
property_name
]
=
""
;
return
addSubForm
({
gadget
:
g
,
property_name
:
value
.
property_name
,
path
:
path
,
type
:
value
.
type
,
schema_path
:
value
.
schema_path
,
schema_part
:
value
.
schema
})
.
push
(
function
(
element
)
{
var
s_e
=
selector
.
element
;
if
(
s_e
)
{
s_e
.
parentNode
.
insertBefore
(
element
,
s_e
);
}
});
},
function
(
gadget_s
,
schema_alternatives
)
{
var
x
,
item_list
=
[[
"
add property
"
,
"
add property
"
]],
item
;
if
(
schema_alternatives
)
{
for
(
x
=
0
;
x
<
schema_alternatives
.
length
;
x
+=
1
)
{
item
=
schema_alternatives
[
x
];
if
(
!
used_properties
.
hasOwnProperty
(
item
.
value
.
property_name
))
{
item_list
.
push
([
item
.
title
,
x
]);
}
}
if
(
gadget_s
)
{
return
{
name
:
gadget_s
.
element
.
getAttribute
(
'
data-gadget-scope
'
),
editable
:
true
,
hidden
:
item_list
.
length
===
1
,
value
:
item_list
[
0
][
1
],
item_list
:
item_list
};
}
return
item_list
.
length
>
1
;
}
});
})
.
push
(
function
(
element
)
{
selector
.
element
=
element
;
return
root_append
(
element
);
})
.
push
(
function
()
{
var
queue
=
RSVP
.
Queue
(),
additionalProperties
;
if
(
json_field
.
patternProperties
!==
undefined
)
{
// XXX need loop on any pattern properties
if
(
json_field
.
patternProperties
[
'
.*
'
]
!==
undefined
)
{
queue
.
push
(
render_object_additionalProperty
.
bind
(
g
,
g
,
"
.* property
"
,
default_dict
,
path
,
json_field
.
patternProperties
[
'
.*
'
],
schema_path
+
'
/patternProperties/.*
'
,
used_properties
,
element_append
))
.
push
(
root_append
);
}
}
if
(
json_field
.
additionalProperties
===
undefined
)
{
additionalProperties
=
true
;
}
else
{
additionalProperties
=
json_field
.
additionalProperties
;
}
if
(
additionalProperties
!==
false
)
{
queue
.
push
(
render_object_additionalProperty
.
bind
(
g
,
g
,
"
additional property
"
,
default_dict
,
path
,
additionalProperties
,
schema_path
+
'
/additionalProperties
'
,
used_properties
,
element_append
))
.
push
(
root_append
);
}
return
queue
;
})
.
push
(
function
()
{
var
key
,
queue
=
RSVP
.
Queue
();
for
(
key
in
default_dict
)
{
if
(
default_dict
.
hasOwnProperty
(
key
))
{
if
(
!
used_properties
.
hasOwnProperty
(
key
))
{
queue
.
push
(
addSubForm
.
bind
(
g
,
{
gadget
:
g
,
property_name
:
key
,
path
:
path
,
schema_path
:
""
,
schema_part
:
undefined
,
default_dict
:
default_dict
[
key
]
})
)
.
push
(
root_append
);
}
}
}
return
queue
;
});
};
function
getFormValuesAsJSONDict
(
g
)
{
var
multi_level_dict
=
{
""
:
{}},
scope
,
options
=
g
.
props
,
array
,
path
,
key
,
i
,
len
,
queue
=
RSVP
.
Queue
();
function
convertOnMultiLevel
(
d
,
key
,
value
)
{
var
ii
,
kk
,
key_list
=
key
.
split
(
"
/
"
);
for
(
ii
=
0
;
ii
<
key_list
.
length
;
ii
+=
1
)
{
kk
=
decodeJsonPointer
(
key_list
[
ii
]);
if
(
ii
===
key_list
.
length
-
1
)
{
if
(
value
!==
undefined
)
{
d
[
kk
]
=
value
;
}
else
{
return
d
[
kk
];
}
}
else
{
if
(
!
d
.
hasOwnProperty
(
kk
))
{
d
[
kk
]
=
{};
}
d
=
d
[
kk
];
}
}
}
function
recursiveGetContent
(
scope
,
path
)
{
queue
.
push
(
function
()
{
return
g
.
getDeclaredGadget
(
scope
);
})
.
push
(
function
(
gadget
)
{
return
gadget
.
getContent
();
})
.
push
(
function
(
jdict
)
{
convertOnMultiLevel
(
multi_level_dict
,
path
,
jdict
);
});
}
function
getContentAndPushArray
(
scope
,
parent_path
)
{
queue
.
push
(
function
()
{
return
g
.
getDeclaredGadget
(
scope
);
})
.
push
(
function
(
gadget
)
{
return
gadget
.
getContent
();
})
.
push
(
function
(
jdict
)
{
var
arr
=
convertOnMultiLevel
(
multi_level_dict
,
parent_path
);
if
(
!
(
arr
instanceof
Array
))
{
arr
=
[];
convertOnMultiLevel
(
multi_level_dict
,
parent_path
,
arr
);
}
arr
.
push
(
jdict
);
});
}
for
(
path
in
options
.
arrays
)
{
if
(
options
.
arrays
.
hasOwnProperty
(
path
))
{
array
=
options
.
arrays
[
path
]
.
querySelectorAll
(
"
div[data-gadget-parent-scope='
"
+
g
.
element
.
getAttribute
(
"
data-gadget-scope
"
)
+
"
']
"
);
len
=
array
.
length
;
for
(
i
=
0
;
i
<
len
;
i
=
i
+
1
)
{
getContentAndPushArray
(
array
[
i
].
getAttribute
(
'
data-gadget-scope
'
),
// slice remove concluding '/'
path
.
slice
(
0
,
-
1
)
);
}
}
}
for
(
path
in
options
.
objects
)
{
if
(
options
.
objects
.
hasOwnProperty
(
path
))
{
for
(
key
in
options
.
objects
[
path
])
{
if
(
options
.
objects
[
path
].
hasOwnProperty
(
key
))
{
scope
=
options
.
objects
[
path
][
key
];
if
(
scope
)
{
recursiveGetContent
(
scope
,
path
+
encodeJsonPointer
(
key
));
}
}
}
}
}
return
queue
.
push
(
function
()
{
var
json_dict
=
{},
k
;
g
.
props
.
inputs
.
forEach
(
function
(
input
)
{
if
(
input
.
name
===
""
||
input
.
value
!==
""
)
{
if
(
input
.
type
===
'
number
'
)
{
json_dict
[
input
.
name
]
=
parseInt
(
input
.
value
,
10
);
}
else
if
(
input
.
value
===
"
true
"
)
{
json_dict
[
input
.
name
]
=
true
;
}
else
if
(
input
.
value
===
"
false
"
)
{
json_dict
[
input
.
name
]
=
false
;
}
else
if
(
input
.
tagName
===
"
TEXTAREA
"
)
{
if
(
input
[
"
data-format
"
]
===
"
string
"
)
{
json_dict
[
input
.
name
]
=
input
.
value
;
}
else
{
json_dict
[
input
.
name
]
=
input
.
value
.
split
(
'
\n
'
);
}
}
else
{
json_dict
[
input
.
name
]
=
input
.
value
;
}
}
});
for
(
k
in
json_dict
)
{
if
(
json_dict
.
hasOwnProperty
(
k
))
{
convertOnMultiLevel
(
multi_level_dict
,
k
,
json_dict
[
k
]);
}
}
return
multi_level_dict
[
""
];
});
}
function
getSubGadgetElement
(
g
,
scope
)
{
return
g
.
element
.
querySelector
(
"
div[data-gadget-scope='
"
+
scope
+
"
']
"
);
}
rJS
(
window
)
.
ready
(
function
()
{
var
g
=
this
;
g
.
props
=
{};
g
.
options
=
{};
})
.
declareAcquiredMethod
(
"
notifyChange
"
,
"
notifyChange
"
)
.
declareAcquiredMethod
(
"
renameChildrenParent
"
,
"
renameChildren
"
)
.
allowPublicAcquisition
(
"
renameChildren
"
,
function
(
opt_arr
,
scope
)
{
var
property_name
,
objects
=
this
.
props
.
objects
,
new_name
=
opt_arr
[
0
],
element
=
getSubGadgetElement
(
this
,
scope
),
parent
=
element
.
getAttribute
(
'
data-json-parent
'
);
if
(
objects
.
hasOwnProperty
(
parent
))
{
parent
=
objects
[
parent
];
if
(
parent
.
hasOwnProperty
(
new_name
))
{
throw
new
Error
(
"
property already exist
"
);
}
// XXX validate property if property pattern
for
(
property_name
in
parent
)
{
if
(
parent
.
hasOwnProperty
(
property_name
)
&&
parent
[
property_name
]
===
scope
)
{
delete
parent
[
property_name
];
parent
[
new_name
]
=
scope
;
return
new_name
;
}
}
throw
new
Error
(
"
gadget not found for renaming
"
);
}
})
.
declareMethod
(
"
rename
"
,
function
(
new_name
,
event
)
{
var
g
=
this
,
name
=
g
.
element
.
getAttribute
(
'
data-json-property-name
'
);
return
this
.
renameChildrenParent
(
new_name
)
.
push
(
function
()
{
return
g
.
element
.
setAttribute
(
'
data-json-property-name
'
,
new_name
);
})
.
push
(
undefined
,
function
(
error
)
{
// XXX notify user
event
.
srcElement
.
value
=
name
;
event
.
srcElement
.
focus
();
});
})
.
declareAcquiredMethod
(
"
selfRemove
"
,
"
deleteChildren
"
)
.
allowPublicAcquisition
(
"
deleteChildren
"
,
function
(
arr
,
scope
)
{
var
g
=
this
,
key
,
i
,
button_list
=
this
.
props
.
add_buttons
,
objects
=
this
.
props
.
objects
,
element
=
getSubGadgetElement
(
g
,
scope
),
parent
=
element
.
getAttribute
(
"
data-json-parent
"
),
tasks
=
[];
if
(
objects
.
hasOwnProperty
(
parent
))
{
parent
=
objects
[
parent
];
for
(
key
in
parent
)
{
if
(
parent
.
hasOwnProperty
(
key
)
&&
parent
[
key
]
===
scope
)
{
delete
parent
[
key
];
}
}
}
element
.
parentNode
.
removeChild
(
element
);
for
(
key
in
g
.
props
.
add_custom_data
)
{
if
(
g
.
props
.
add_custom_data
.
hasOwnProperty
(
key
))
{
tasks
.
push
(
g
.
props
.
add_custom_data
[
key
].
rerender
());
}
}
for
(
i
=
0
;
i
<
button_list
.
length
;
i
=
i
+
1
)
{
tasks
.
push
(
button_list
[
i
].
rerender
());
}
tasks
.
push
(
checkValidityAndNotifyChange
(
g
));
return
RSVP
.
Queue
()
.
push
(
function
()
{
return
RSVP
.
all
(
tasks
);
});
})
.
declareMethod
(
'
getElementByPath
'
,
function
(
data_path
)
{
var
g
=
this
,
array
,
path
,
scope
,
key
,
slash_count
=
0
,
slash_count_next
,
bingo
,
idx
,
options
=
g
.
props
;
if
(
data_path
!==
"
/
"
)
{
for
(
path
in
options
.
arrays
)
{
if
(
options
.
arrays
.
hasOwnProperty
(
path
)
&&
data_path
.
startsWith
(
path
))
{
array
=
options
.
arrays
[
path
]
.
querySelectorAll
(
"
div[data-gadget-parent-scope='
"
+
g
.
element
.
getAttribute
(
"
data-gadget-scope
"
)
+
"
']
"
);
data_path
=
data_path
.
slice
(
path
.
length
).
split
(
"
/
"
);
idx
=
data_path
[
0
];
data_path
=
"
/
"
+
data_path
.
slice
(
1
).
join
(
"
/
"
);
bingo
=
array
[
idx
].
getAttribute
(
'
data-gadget-scope
'
);
}
}
if
(
bingo
)
{
return
g
.
getDeclaredGadget
(
bingo
)
.
push
(
function
(
gadget
)
{
return
gadget
.
getElementByPath
(
data_path
);
});
}
for
(
path
in
options
.
objects
)
{
if
(
options
.
objects
.
hasOwnProperty
(
path
)
&&
data_path
.
startsWith
(
path
))
{
slash_count_next
=
path
.
split
(
"
/
"
).
length
-
1
;
if
(
slash_count_next
>
slash_count
)
{
bingo
=
path
;
slash_count
=
slash_count_next
;
}
}
}
if
(
bingo
)
{
path
=
options
.
objects
[
bingo
];
for
(
key
in
path
)
{
if
(
path
.
hasOwnProperty
(
key
))
{
if
(
data_path
.
startsWith
(
bingo
+
encodeJsonPointer
(
key
)))
{
data_path
=
data_path
.
slice
(
bingo
.
length
+
encodeJsonPointer
(
key
).
length
);
if
(
!
data_path
)
{
data_path
=
"
/
"
;
}
scope
=
path
[
key
];
}
}
}
}
if
(
scope
)
{
return
g
.
getDeclaredGadget
(
scope
)
.
push
(
function
(
gadget
)
{
return
gadget
.
getElementByPath
(
data_path
);
});
}
}
return
RSVP
.
Queue
()
.
push
(
function
()
{
return
document
.
getElementById
(
g
.
element
.
getAttribute
(
"
data-gadget-scope
"
)
+
data_path
);
});
})
.
declareAcquiredMethod
(
"
notifyValid
"
,
"
notifyValid
"
)
.
declareAcquiredMethod
(
"
notifyInvalid
"
,
"
notifyInvalid
"
)
.
declareAcquiredMethod
(
"
checkValidity
"
,
"
checkValidity
"
)
.
allowPublicAcquisition
(
"
notifyValid
"
,
function
(
arr
,
sub_scope
)
{
return
true
;
})
.
allowPublicAcquisition
(
"
notifyChange
"
,
function
(
arr
,
sub_scope
)
{
var
g
=
this
,
opt
=
arr
[
0
],
event_object
;
event_object
=
g
.
props
.
add_custom_data
[
sub_scope
];
if
(
event_object
&&
opt
.
type
===
"
change
"
)
{
return
event_object
.
event
();
}
return
g
.
notifyChange
();
})
.
declareMethod
(
'
renderForm
'
,
function
(
options
)
{
var
g
=
this
,
property_name
=
g
.
element
.
getAttribute
(
'
data-json-property-name
'
),
schema
=
options
.
schema
,
delete_button
,
root
;
g
.
props
.
inputs
=
[];
g
.
props
.
add_buttons
=
[];
g
.
props
.
add_custom_data
=
{};
g
.
props
.
arrays
=
{};
g
.
props
.
objects
=
{};
g
.
props
.
path
=
options
.
path
;
// self gadget scope
if
(
!
property_name
||
!
options
.
display_label
)
{
property_name
=
""
;
}
root
=
g
.
element
.
querySelector
(
'
[data-json-path="/"]
'
);
if
(
!
root
)
{
root
=
g
.
element
;
}
while
(
root
.
firstChild
)
{
root
.
removeChild
(
root
.
firstChild
);
}
if
(
!
options
.
top
)
{
delete_button
=
document
.
createElement
(
"
button
"
);
delete_button
.
setAttribute
(
"
class
"
,
"
ui-shadow-inset ui-btn ui-btn-inline ui-corner-all
"
+
"
ui-btn-icon-notext ui-icon-btn ui-icon-trash-o ui-input-btn
"
);
delete_button
.
type
=
"
button
"
;
delete_button
.
name
=
options
.
path
;
g
.
props
.
delete_button
=
delete_button
;
root
.
appendChild
(
delete_button
);
}
return
render_field
(
g
,
property_name
,
""
,
schema
,
options
.
document
,
root
,
options
.
schema_path
,
{
type
:
options
.
type
})
.
push
(
function
()
{
g
.
listenEvents
();
return
g
.
element
;
});
})
.
declareAcquiredMethod
(
"
expandSchema
"
,
"
expandSchema
"
)
.
onEvent
(
'
click
'
,
function
(
evt
)
{
if
(
evt
.
target
===
this
.
props
.
delete_button
)
{
return
this
.
selfRemove
(
evt
);
}
var
link
=
evt
.
target
.
getAttribute
(
"
data-error-link
"
),
button_list
=
this
.
props
.
add_buttons
,
i
;
if
(
link
)
{
location
.
href
=
link
;
return
;
}
for
(
i
=
0
;
i
<
button_list
.
length
;
i
=
i
+
1
)
{
if
(
evt
.
target
===
button_list
[
i
].
element
)
{
return
button_list
[
i
].
event
(
evt
);
}
}
})
.
onEvent
(
'
change
'
,
function
(
evt
)
{
if
(
evt
.
target
===
this
.
props
.
property_name_edit
)
{
return
this
.
rename
(
this
.
props
.
property_name_edit
.
value
,
evt
);
}
var
field_list
=
this
.
props
.
inputs
,
i
;
for
(
i
=
0
;
i
<
field_list
.
length
;
i
=
i
+
1
)
{
if
(
evt
.
target
===
field_list
[
i
])
{
return
checkValidityAndNotifyChange
(
this
);
}
}
})
.
declareJob
(
'
listenEvents
'
,
function
()
{
// XXX Disable
return
;
})
.
declareMethod
(
'
getContent
'
,
function
()
{
var
g
=
this
;
return
getFormValuesAsJSONDict
(
g
);
});
}(
window
,
document
,
location
,
rJS
,
RSVP
,
jIO
,
tv4
));
\ No newline at end of file
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment