grails.databinding.SimpleDataBinder.groovy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of grace-databinding Show documentation
Show all versions of grace-databinding Show documentation
Grace Framework : Grace Databinding
/*
* Copyright 2014-2023 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package grails.databinding
import java.lang.annotation.Annotation
import java.lang.reflect.Array
import java.lang.reflect.Field
import java.lang.reflect.Modifier
import java.lang.reflect.ParameterizedType
import java.lang.reflect.Type
import java.security.ProtectionDomain
import java.util.regex.Matcher
import groovy.transform.CompileStatic
import groovy.transform.TypeCheckingMode
import groovy.xml.slurpersupport.GPathResult
import grails.databinding.converters.FormattedValueConverter
import grails.databinding.converters.ValueConverter
import grails.databinding.events.DataBindingListener
import grails.databinding.initializers.ValueInitializer
import org.grails.databinding.ClosureValueConverter
import org.grails.databinding.ClosureValueInitializer
import org.grails.databinding.IndexedPropertyReferenceDescriptor
import org.grails.databinding.converters.ConversionService
import org.grails.databinding.converters.FormattedDateValueConverter
import org.grails.databinding.converters.StructuredCalendarBindingEditor
import org.grails.databinding.converters.StructuredDateBindingEditor
import org.grails.databinding.converters.StructuredSqlDateBindingEditor
import org.grails.databinding.errors.SimpleBindingError
import org.grails.databinding.xml.GPathResultMap
/**
* A data binder that will bind nested Maps to an object.
*
class Person {
String firstName
Address homeAddress
}
class Address {
String city
String state
}
def person = new Person()
def binder = new SimpleDataBinder()
binder.bind person, [firstName: 'Steven', homeAddress: [city: 'St. Louis', state: 'Missouri']]
assert person.firstName == 'Steven'
assert person.homeAddress.city == 'St. Louis'
assert person.homeAddress.state == 'Missouri'
*
* @author Jeff Brown
* @author Graeme Rocher
*
* @since 3.0
*/
@CompileStatic
class SimpleDataBinder implements DataBinder {
protected Map structuredEditors = new HashMap()
ConversionService conversionService
protected Map> conversionHelpers = [:].withDefault { c -> [] }
protected Map formattedValueConversionHelpers = new HashMap()
protected static final List BASIC_TYPES = [
String,
Boolean,
Byte,
Short,
Integer,
Long,
Float,
Double,
Character
] as List
static final INDEXED_PROPERTY_REGEX = /(.*)\[\s*([^\s]*)\s*\]\s*$/
int autoGrowCollectionLimit = 256
SimpleDataBinder() {
registerStructuredEditor Date, new StructuredDateBindingEditor()
registerStructuredEditor java.sql.Date, new StructuredSqlDateBindingEditor()
registerStructuredEditor Calendar, new StructuredCalendarBindingEditor()
registerFormattedValueConverter new FormattedDateValueConverter()
}
void registerStructuredEditor(Class clazz, StructuredBindingEditor editor) {
structuredEditors[clazz] = editor
}
void registerConverter(ValueConverter converter) {
conversionHelpers[converter.targetType] << converter
}
void registerFormattedValueConverter(FormattedValueConverter converter) {
formattedValueConversionHelpers[converter.targetType] = converter
}
/**
*
* @param obj The object being bound to
* @param source The data binding source
* @see DataBindingSource
*/
@Override
void bind(Object obj, DataBindingSource source) {
bind(obj, source, null, null, null, null)
}
/**
*
* @param obj The object being bound to
* @param source The data binding source
* @param listener A listener which will be notified of data binding events triggered
* by this binding
* @see DataBindingSource
* @see DataBindingListener
*/
@Override
void bind(Object obj, DataBindingSource source, DataBindingListener listener) {
bind(obj, source, null, null, null, listener)
}
/**
*
* @param obj The object being bound to
* @param source The data binding source
* @param whiteList A list of property names to be included during this data binding.
* All other properties represented in the binding source will be ignored.
* @see DataBindingSource
*/
@Override
void bind(Object obj, DataBindingSource source, List whiteList) {
bind(obj, source, null, whiteList, null, null)
}
/**
*
* @param obj The object being bound to
* @param source The data binding source
* @param whiteList A list of property names to be included during this data binding.
* All other properties represented in the binding source will be ignored
* @param blackList A list of properties names to be excluded during this data binding.
* @see DataBindingSource
*/
@Override
void bind(Object obj, DataBindingSource source, List whiteList, List blackList) {
bind(obj, source, null, whiteList, blackList, null)
}
/**
*
* @param obj The object being bound to
* @param gpath A GPathResult which represents the data being bound.
* @see DataBindingSource
*/
@Override
void bind(Object obj, GPathResult gpath) {
bind(obj, new SimpleMapDataBindingSource(new GPathResultMap(gpath)))
}
/**
*
* @param obj The object being bound to
* @param source The data binding source
* @param filter Only properties beginning with filter will be included in the
* data binding. For example, if filter is "person" and the binding
* source contains data for properties "person.name" and "author.name"
* the value of "person.name" will be bound to obj.name. The value of
* "author.name" will be ignored.
* @param whiteList A list of property names to be included during this data binding.
* All other properties represented in the binding source will be ignored
* @param blackList A list of properties names to be excluded during this data binding.
* @see DataBindingSource
*/
@Override
void bind(Object obj, DataBindingSource source, String filter, List whiteList, List blackList) {
bind(obj, source, filter, whiteList, blackList, null)
}
/**
*
* @param obj The object being bound to
* @param source The data binding source
* @param filter Only properties beginning with filter will be included in the
* data binding. For example, if filter is "person" and the binding
* source contains data for properties "person.name" and "author.name"
* the value of "person.name" will be bound to obj.name.
* The value of "author.name" will be ignored.
* @param whiteList A list of property names to be included during this data binding.
* All other properties represented in the binding source will be ignored
* @param blackList A list of properties names to be excluded during this data binding.
* @param listener A listener which will be notified of data binding events triggered by this binding
* @see DataBindingSource
* @see DataBindingListener
*/
@Override
void bind(Object obj, DataBindingSource source, String filter, List whiteList, List blackList, DataBindingListener listener) {
doBind(obj, source, filter, whiteList, blackList, listener, null)
}
protected void doBind(Object obj, DataBindingSource source, String filter, List whiteList, List blackList,
DataBindingListener listener, Object errors) {
Set keys = source.getPropertyNames()
for (String key in keys) {
if (!filter || key.startsWith(filter + '.')) {
String propName = key
if (filter) {
propName = key[(1 + filter.size())..-1]
}
MetaProperty metaProperty = obj.metaClass.getMetaProperty propName
if (metaProperty) { // normal property
if (isOkToBind(metaProperty, whiteList, blackList)) {
Object val = source[key]
try {
ValueConverter converter = getValueConverter(obj, metaProperty.name)
if (converter) {
bindProperty(obj, source, metaProperty, converter.convert(source), listener, errors)
}
else {
processProperty(obj, metaProperty, preprocessValue(val), source, listener, errors)
}
}
catch (Exception e) {
addBindingError(obj, propName, val, e, listener, errors)
}
}
}
else {
IndexedPropertyReferenceDescriptor descriptor = getIndexedPropertyReferenceDescriptor propName
if (descriptor) { // indexed property
metaProperty = obj.metaClass.getMetaProperty descriptor.propertyName
if (metaProperty && isOkToBind(metaProperty, whiteList, blackList)) {
Object val = source.getPropertyValue(key)
processIndexedProperty(obj, metaProperty, descriptor, val, source, listener, errors)
}
}
else if (propName.startsWith('_') && propName.length() > 1) { // boolean special handling
String restOfPropertyName = propName[1..-1]
if (!source.containsProperty(restOfPropertyName)) {
metaProperty = obj.metaClass.getMetaProperty restOfPropertyName
if (metaProperty && isOkToBind(metaProperty, whiteList, blackList)) {
if ((Boolean == metaProperty.type || Boolean.TYPE == metaProperty.type)) {
bindProperty(obj, source, metaProperty, false, listener, errors)
}
}
}
}
}
}
}
}
protected boolean isOkToBind(String propName, List whiteList, List blackList) {
'class' != propName && 'classLoader' != propName && 'protectionDomain' != propName && 'metaClass' != propName
&& !blackList?.contains(propName)
&& (!whiteList || whiteList.contains(propName) || whiteList.find { it -> it?.toString()?.startsWith(propName + '.') })
}
protected boolean isOkToBind(MetaProperty property, List whitelist, List blacklist) {
isOkToBind(property.name, whitelist, blacklist)
&& (property.type != null && !(ClassLoader.isAssignableFrom(property.type)
|| ProtectionDomain.isAssignableFrom(property.type)))
}
protected IndexedPropertyReferenceDescriptor getIndexedPropertyReferenceDescriptor(String propName) {
IndexedPropertyReferenceDescriptor descriptor = null
Matcher matcher = (propName =~ INDEXED_PROPERTY_REGEX)
if (matcher) {
String indexedPropertyName = matcher.group(1)
String index = matcher.group(2)
if (index.size() > 2 && ((index.startsWith("'") && index.endsWith("'")) || (index.startsWith('"') && index.endsWith('"')))) {
index = index[1..-2]
}
descriptor = new IndexedPropertyReferenceDescriptor(propertyName: indexedPropertyName, index: index)
}
descriptor
}
protected void processProperty(Object obj, MetaProperty metaProperty, Object val,
DataBindingSource source, DataBindingListener listener, Object errors) {
String propName = metaProperty.name
Class propertyType = metaProperty.type
if (structuredEditors.containsKey(propertyType) && (val == 'struct' || val == 'date.struct')) {
StructuredBindingEditor structuredEditor = structuredEditors[propertyType]
val = structuredEditor.getPropertyValue(obj, propName, source)
}
bindProperty(obj, source, metaProperty, val, listener, errors)
}
protected SimpleMapDataBindingSource splitIndexedStruct(IndexedPropertyReferenceDescriptor indexedPropertyReferenceDescriptor,
DataBindingSource source) {
String propName = indexedPropertyReferenceDescriptor.propertyName
Map structValues = new HashMap()
String prefix = indexedPropertyReferenceDescriptor
for (String propertyName : source.propertyNames) {
if (propertyName.startsWith(prefix)) {
String deIndexedPropertyName = propName
String[] parts = propertyName.split('_')
if (parts.length > 1) {
deIndexedPropertyName = deIndexedPropertyName + '_' + parts[1]
}
structValues.put(deIndexedPropertyName, source.getPropertyValue(propertyName))
}
}
new SimpleMapDataBindingSource(structValues)
}
protected void processIndexedProperty(Object obj, MetaProperty metaProperty,
IndexedPropertyReferenceDescriptor indexedPropertyReferenceDescriptor,
Object val, DataBindingSource source, DataBindingListener listener, Object errors) {
String propName = indexedPropertyReferenceDescriptor.propertyName
Class propertyType = metaProperty.type
Class genericType = getReferencedTypeForCollection(propName, obj)
if (structuredEditors.containsKey(genericType) && (val == 'struct' || val == 'date.struct')) {
StructuredBindingEditor structuredEditor = structuredEditors[genericType]
val = structuredEditor.getPropertyValue(obj, propName, splitIndexedStruct(indexedPropertyReferenceDescriptor, source))
}
if (propertyType.isArray()) {
int index = Integer.parseInt(indexedPropertyReferenceDescriptor.index)
Object[] array = initializeArray(obj, propName, propertyType.componentType, index)
if (array != null) {
addElementToArrayAt(array, index, val)
}
}
else if (Collection.isAssignableFrom(propertyType)) {
int index = Integer.parseInt(indexedPropertyReferenceDescriptor.index)
Collection collectionInstance = initializeCollection(obj, propName, propertyType)
Object indexedInstance = null
if (!(Set.isAssignableFrom(propertyType))) {
indexedInstance = collectionInstance[index]
}
if (indexedInstance == null) {
if (genericType) {
if (genericType.isAssignableFrom(val?.getClass())) {
addElementToCollectionAt(obj, propName, collectionInstance, index, val)
}
else if (isBasicType(genericType)) {
addElementToCollectionAt(obj, propName, collectionInstance, index, convert(genericType, val))
}
else if (val instanceof Map) {
indexedInstance = genericType.newInstance()
bind(indexedInstance, new SimpleMapDataBindingSource(val), listener)
addElementToCollectionAt(obj, propName, collectionInstance, index, indexedInstance)
}
else if (val instanceof DataBindingSource) {
indexedInstance = genericType.newInstance()
bind(indexedInstance, val, listener)
addElementToCollectionAt(obj, propName, collectionInstance, index, indexedInstance)
}
else if (genericType.isEnum() && val instanceof CharSequence) {
Object enumValue = convertStringToEnum(genericType, val.toString())
addElementToCollectionAt(obj, propName, collectionInstance, index, enumValue)
}
else {
addElementToCollectionAt(obj, propName, collectionInstance, index, convert(genericType, val))
}
}
else {
addElementToCollectionAt(obj, propName, collectionInstance, index, val)
}
}
else {
if (val instanceof Map) {
bind(indexedInstance, new SimpleMapDataBindingSource(val), listener)
}
else if (val instanceof DataBindingSource) {
bind(indexedInstance, val, listener)
}
else if (val == null && indexedInstance != null) {
addElementToCollectionAt(obj, propName, collectionInstance, index, null)
}
}
}
else if (Map.isAssignableFrom(propertyType)) {
Map mapInstance = initializeMap(obj, propName)
if (mapInstance.size() < autoGrowCollectionLimit || mapInstance.containsKey(indexedPropertyReferenceDescriptor.index)) {
Class referencedType = getReferencedTypeForCollection(propName, obj)
if (referencedType != null) {
if (val instanceof Map) {
mapInstance[indexedPropertyReferenceDescriptor.index] = referencedType.newInstance(val)
}
else {
mapInstance[indexedPropertyReferenceDescriptor.index] = convert(referencedType, val)
}
}
else {
mapInstance[indexedPropertyReferenceDescriptor.index] = val
}
}
}
}
@CompileStatic(TypeCheckingMode.SKIP)
protected Object[] initializeArray(obj, String propertyName, Class arrayType, int index) {
Object[] array = obj[propertyName]
if (array == null && index < autoGrowCollectionLimit) {
array = Array.newInstance(arrayType, index + 1)
obj[propertyName] = array
}
else if (array != null && array.length <= index && index < autoGrowCollectionLimit) {
Object newArray = Array.newInstance(arrayType, index + 1)
System.arraycopy(array, 0, newArray, 0, array.length)
array = newArray
obj[propertyName] = newArray
}
array
}
protected boolean isBasicType(Class c) {
BASIC_TYPES.contains(c) || c.isPrimitive()
}
protected Class> getReferencedTypeForCollectionInClass(String propertyName, Class clazz) {
Class referencedType
Field field = getField(clazz, propertyName)
if (field) {
Type genericType = field.genericType
if (genericType instanceof ParameterizedType) {
ParameterizedType pt = (ParameterizedType) genericType
Class rawType = pt.getRawType()
if (Map.isAssignableFrom(rawType)) {
referencedType = pt.getActualTypeArguments()[1]
}
else {
referencedType = pt.getActualTypeArguments()[0]
}
}
}
referencedType
}
protected Class> getReferencedTypeForCollection(String propertyName, Object obj) {
getReferencedTypeForCollectionInClass(propertyName, obj.getClass())
}
protected boolean isOkToAddElementAt(Collection collection, int index) {
boolean isOk
if (collection instanceof Set) {
isOk = collection.size() < autoGrowCollectionLimit
}
else {
isOk = (index < autoGrowCollectionLimit || index < collection.size())
}
isOk
}
@CompileStatic(TypeCheckingMode.SKIP)
protected void addElementToCollectionAt(Object obj, String propertyName, Collection collection, int index, Object val) {
if (isOkToAddElementAt(collection, index)) {
if (collection instanceof Set) {
collection.add(val)
}
else {
collection[index] = val
}
}
}
@CompileStatic(TypeCheckingMode.SKIP)
protected addElementToArrayAt(Object[] array, int index, Object val) {
if (array.length > index) {
array[index] = convert(array.class.componentType, val)
}
}
protected Map initializeMap(Object obj, String propertyName) {
if (obj[propertyName] == null) {
obj[propertyName] = [:]
}
(Map) obj[propertyName]
}
protected Collection initializeCollection(Object obj, String propertyName, Class type, boolean reuseExistingCollectionIfExists = true) {
Object val = null
if (reuseExistingCollectionIfExists) {
val = obj[propertyName]
}
if (val == null) {
val = getDefaultCollectionInstanceForType(type)
obj[propertyName] = val
}
(Collection) val
}
protected Object getDefaultCollectionInstanceForType(Class type) {
Object val
if (List.isAssignableFrom(type)) {
val = []
}
else if (SortedSet.isAssignableFrom(type)) {
val = new TreeSet()
}
else if (LinkedHashSet.isAssignableFrom(type)) {
val = new LinkedHashSet()
}
else if (Collection.isAssignableFrom(type)) {
val = new HashSet()
}
val
}
/**
* Get a ValueConverter for field
*
* @param field The field to retrieve a converter for
* @param formattingValue The format that the converter will use
* @return a ValueConverter for field which uses formattingValue for its format
* @see BindingFormat
*/
protected ValueConverter getFormattedConverter(Field field, String formattingValue) {
ValueConverter converter
FormattedValueConverter formattedConverter = formattedValueConversionHelpers[field.type]
if (formattedConverter) {
converter = { SimpleMapDataBindingSource source ->
Object value = preprocessValue(source.getPropertyValue(field.name))
Object convertedValue = null
if (value != null) {
convertedValue = formattedConverter.convert(value, formattingValue)
}
convertedValue
} as ValueConverter
}
converter
}
protected Field getField(Class clazz, String fieldName) {
Field field = null
try {
field = clazz.getDeclaredField(fieldName)
}
catch (NoSuchFieldException ignored) {
Class superClass = clazz.getSuperclass()
if (superClass != Object) {
field = getField(superClass, fieldName)
}
}
field
}
protected ValueConverter getValueConverterForField(Object obj, String propName) {
ValueConverter converter
try {
Field field = getField(obj.getClass(), propName)
if (field) {
Annotation annotation = field.getAnnotation(BindUsing)
if (annotation) {
Class valueClass = getValueOfBindUsing(annotation)
if (Closure.isAssignableFrom(valueClass)) {
Closure closure = (Closure) valueClass.newInstance(null, null)
converter = new ClosureValueConverter(converterClosure: closure.curry(obj), targetType: field.type)
}
}
else {
annotation = field.getAnnotation(BindingFormat)
if (annotation) {
converter = getFormattedConverter(field, getFormatString(annotation))
}
}
}
}
catch (Exception ignored) {
}
converter
}
/**
* @param annotation An instance of grails.databinding.BindingUsing or org.grails.databinding.BindingUsing
* @return the value Class of the annotation
*/
protected Class getValueOfBindUsing(Annotation annotation) {
assert annotation instanceof BindUsing
Class value
if (annotation instanceof BindUsing) {
value = ((BindUsing) annotation).value()
}
value
}
/**
* @param annotation An instance of grails.databinding.BindingFormat or org.grails.databinding.BindingFormat
* @return the value String of the annotation
*/
protected String getFormatString(Annotation annotation) {
assert annotation instanceof BindingFormat
String formatString
if (annotation instanceof BindingFormat) {
formatString = ((BindingFormat) annotation).value()
}
formatString
}
protected ValueConverter getValueConverterForClass(Object obj, String propName) {
ValueConverter converter
Class objClass = obj.getClass()
Annotation annotation = objClass.getAnnotation(BindUsing)
if (annotation) {
Class valueClass = getValueOfBindUsing(annotation)
if (BindingHelper.isAssignableFrom(valueClass)) {
BindingHelper dataConverter = (BindingHelper) valueClass.newInstance()
converter = new ClosureValueConverter(converterClosure: { DataBindingSource it -> dataConverter.getPropertyValue(obj, propName, it) })
}
}
converter
}
protected ValueConverter getValueConverter(Object obj, String propName) {
ValueConverter converter = getValueConverterForField obj, propName
converter = converter ?: getValueConverterForClass(obj, propName)
converter
}
@CompileStatic(TypeCheckingMode.SKIP)
protected convertStringToEnum(Class extends Enum> enumClass, String value) {
try {
enumClass.valueOf(value)
}
catch (IllegalArgumentException ignored) {
}
}
protected Object preprocessValue(Object propertyValue) {
propertyValue
}
protected void setPropertyValue(Object obj, DataBindingSource source, MetaProperty metaProperty,
Object propertyValue, DataBindingListener listener) {
boolean convertCollectionElements = false
if (propertyValue instanceof Collection) {
Class referencedType = getReferencedTypeForCollection(metaProperty.name, obj)
if (referencedType) {
Object nonAssignableValue = propertyValue.find { it != null && !(referencedType.isAssignableFrom(it.getClass())) }
if (nonAssignableValue != null) {
convertCollectionElements = true
}
}
}
setPropertyValue(obj, source, metaProperty, propertyValue, listener, convertCollectionElements)
}
protected void setPropertyValue(Object obj, DataBindingSource source, MetaProperty metaProperty,
Object propertyValue, DataBindingListener listener, boolean convertCollectionElements) {
String propName = metaProperty.name
Type propertyType
MetaMethod propertyGetter
if (metaProperty instanceof MetaBeanProperty) {
MetaBeanProperty mbp = (MetaBeanProperty) metaProperty
propertyType = mbp.getter?.returnType ?: mbp.field?.type
if (propertyType && (propertyType.interface || Modifier.isAbstract(propertyType.modifiers))) {
propertyType = mbp.field?.type
}
propertyGetter = mbp.getter
}
if (propertyType == null || propertyType == Object) {
propertyType = metaProperty.type
if (propertyType == null || propertyType == Object) {
propertyType = getField(obj.getClass(), propName)?.type ?: Object
}
}
if (propertyValue == null || propertyType == Object || propertyType.isAssignableFrom(propertyValue.getClass())) {
if (convertCollectionElements && ((!(propertyValue instanceof Range) && propertyValue instanceof Collection &&
Collection.isAssignableFrom(propertyType) && propertyGetter))) {
addElementsToCollection(obj, propName, propertyValue, true)
}
else {
obj[propName] = propertyValue
}
}
else if (propertyValue instanceof List &&
Set.isAssignableFrom(propertyType) &&
!SortedSet.isAssignableFrom(propertyType)) {
addElementsToCollection(obj, propName, propertyValue, true)
}
else {
if (propertyValue instanceof Map) {
if (Collection.isAssignableFrom(propertyType) &&
propertyValue.size() == 1 &&
((Map) propertyValue)[propertyValue.keySet()[0]] instanceof List) {
Object key = propertyValue.keySet()[0]
List list = (List) ((Map) propertyValue)[key]
addElementsToCollection(obj, propName, list)
}
else {
if (obj[propName] == null) {
initializeProperty(obj, propName, propertyType, source)
}
bind(obj[propName], new SimpleMapDataBindingSource(propertyValue), listener)
}
}
else if (propertyValue instanceof DataBindingSource) {
if (Collection.isAssignableFrom(propertyType) &&
propertyValue.size() == 1 &&
((Map) propertyValue)[propertyValue.getPropertyNames()[0]] instanceof List) {
String key = propertyValue.getPropertyNames()[0]
List list = (List) ((Map) propertyValue)[key]
addElementsToCollection(obj, propName, list)
}
else {
if (obj[propName] == null) {
initializeProperty(obj, propName, propertyType, source)
}
bind(obj[propName], propertyValue, listener)
}
}
else if (Collection.isAssignableFrom(propertyType) && propertyValue instanceof String) {
addElementToCollection(obj, propName, propertyType, propertyValue, true)
}
else if (Collection.isAssignableFrom(propertyType) && propertyValue instanceof Number) {
addElementToCollection(obj, propName, propertyType, propertyValue, true)
}
else if (Collection.isAssignableFrom(propertyType) && propertyValue.getClass().isArray()) {
addElementsToCollection(obj, propName, propertyValue as Collection, true)
}
else {
obj[propName] = convert(propertyType, propertyValue)
}
}
}
protected boolean addElementToCollection(Object obj, String propName, Class propertyType, Object propertyValue, boolean clearCollection) {
boolean isSet = false
Collection coll = initializeCollection(obj, propName, propertyType)
if (coll != null) {
if (clearCollection) {
coll.clear()
}
Class referencedType = getReferencedTypeForCollection propName, obj
if (referencedType != null) {
if (propertyValue == null || referencedType.isAssignableFrom(propertyValue.getClass())) {
coll << propertyValue
isSet = true
}
else {
coll << convert(referencedType, propertyValue)
isSet = true
}
}
}
isSet
}
protected bindProperty(Object obj, DataBindingSource source, MetaProperty metaProperty,
Object propertyValue, DataBindingListener listener, Object errors) {
String propName = metaProperty.name
if (listener == null || listener.beforeBinding(obj, propName, propertyValue, errors) != false) {
try {
setPropertyValue(obj, source, metaProperty, propertyValue, listener)
}
catch (Exception e) {
addBindingError(obj, propName, propertyValue, e, listener, errors)
}
}
else if (listener != null && propertyValue instanceof Map && obj[propName] != null) {
bind(obj[propName], new SimpleMapDataBindingSource(propertyValue))
}
listener?.afterBinding(obj, propName, errors)
}
protected void addBindingError(Object obj, String propName, propertyValue, Exception e, DataBindingListener listener, Object errors) {
if (listener) {
SimpleBindingError error = new SimpleBindingError(obj, propName, propertyValue, e.cause ?: e)
listener.bindingError error, errors
}
}
private void addElementsToCollection(Object obj, String collectionPropertyName, Collection collection, boolean removeExistingElements = false) {
Class propertyType = obj.metaClass.getMetaProperty(collectionPropertyName).type
Class referencedType = getReferencedTypeForCollection(collectionPropertyName, obj)
Collection coll = initializeCollection(obj, collectionPropertyName, propertyType, !removeExistingElements)
if (removeExistingElements) {
coll.clear()
}
for (element in collection) {
if (element == null || referencedType == null || referencedType.isAssignableFrom(element.getClass())) {
coll << element
}
else {
coll << convert(referencedType, element)
}
}
obj[collectionPropertyName] = coll
}
protected void initializeProperty(Object obj, String propName, Class propertyType, DataBindingSource source) {
ValueInitializer initializer = getPropertyInitializer(obj, propName)
if (initializer) {
obj[propName] = initializer.initialize()
}
else {
obj[propName] = propertyType.newInstance()
}
}
protected ValueInitializer getPropertyInitializer(Object obj, String propName) {
ValueInitializer initializer = getValueInitializerForField obj, propName
initializer
}
protected ValueInitializer getValueInitializerForField(Object obj, String propName) {
ValueInitializer initializer
try {
Field field = getField(obj.getClass(), propName)
if (field) {
Annotation annotation = field.getAnnotation(BindInitializer)
if (annotation) {
Class valueClass = getValueOfBindInitializer(annotation)
if (Closure.isAssignableFrom(valueClass)) {
Closure closure = (Closure) valueClass.newInstance(null, null)
initializer = new ClosureValueInitializer(initializerClosure: closure.curry(obj), targetType: field.type)
}
}
}
}
catch (Exception ignored) {
}
initializer
}
/**
* @param annotation An instance of grails.databinding.BindInitializer
* @return the value Class of the annotation
*/
protected Class getValueOfBindInitializer(Annotation annotation) {
assert annotation instanceof BindInitializer
Class value
if (annotation instanceof BindInitializer) {
value = ((BindInitializer) annotation).value()
}
value
}
protected Object convert(Class typeToConvertTo, Object value) {
if (value == null || typeToConvertTo.isAssignableFrom(value?.getClass())) {
return value
}
if (conversionHelpers.containsKey(typeToConvertTo)) {
ValueConverter converter = getConverter(typeToConvertTo, value)
if (converter) {
return converter.convert(value)
}
}
if (conversionService?.canConvert(value.getClass(), typeToConvertTo)) {
return conversionService.convert(value, typeToConvertTo)
}
if (Collection.isAssignableFrom(typeToConvertTo) && value instanceof String[]) {
if (typeToConvertTo == Set) {
return value as Set
}
if (typeToConvertTo == List) {
return value as List
}
}
else if (typeToConvertTo.isPrimitive() || typeToConvertTo.isArray()) {
return value
}
else if (value instanceof Map) {
Object obj = typeToConvertTo.newInstance()
bind(obj, new SimpleMapDataBindingSource(value))
return obj
}
else if (Enum.isAssignableFrom(typeToConvertTo) && value instanceof String) {
return convertStringToEnum((Class extends Enum>) typeToConvertTo, value)
}
typeToConvertTo.newInstance(value)
}
protected ValueConverter getConverter(Class typeToConvertTo, Object value) {
List converters = conversionHelpers.get(typeToConvertTo)
converters?.find { ValueConverter c -> c.canConvert(value) }
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy