All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.qt.internal.AbstractMetaObjectUtility Maven / Gradle / Ivy

/****************************************************************************
**
** Copyright (C) 2009-2024 Dr. Peter Droste, Omix Visualization GmbH & Co. KG. All rights reserved.
**
** This file is part of Qt Jambi.
**
** ** $BEGIN_LICENSE$
** GNU Lesser General Public License Usage
** This file may be used under the terms of the GNU Lesser
** General Public License version 2.1 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPL included in the
** packaging of this file.  Please review the following information to
** ensure the GNU Lesser General Public License version 2.1 requirements
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
** 
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 3.0 as published by the Free Software
** Foundation and appearing in the file LICENSE.GPL included in the
** packaging of this file.  Please review the following information to
** ensure the GNU General Public License version 3.0 requirements will be
** met: http://www.gnu.org/copyleft/gpl.html.
** $END_LICENSE$
**
** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
**
****************************************************************************/
package io.qt.internal;

import java.lang.reflect.AnnotatedElement;
import java.lang.reflect.AnnotatedParameterizedType;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.function.BiFunction;

import io.qt.Nullable;
import io.qt.QPropertyDeclarationException;
import io.qt.QtObject;
import io.qt.QtPointerType;
import io.qt.QtPrimitiveType;
import io.qt.QtPropertyBindable;
import io.qt.QtReferenceType;
import io.qt.core.QBindable;
import io.qt.core.QBooleanBindable;
import io.qt.core.QByteBindable;
import io.qt.core.QCharBindable;
import io.qt.core.QDoubleBindable;
import io.qt.core.QFloatBindable;
import io.qt.core.QIntBindable;
import io.qt.core.QList;
import io.qt.core.QLongBindable;
import io.qt.core.QMetaObject;
import io.qt.core.QMetaProperty;
import io.qt.core.QMetaType;
import io.qt.core.QObject;
import io.qt.core.QQueue;
import io.qt.core.QSet;
import io.qt.core.QShortBindable;
import io.qt.core.QStack;
import io.qt.core.QUntypedBindable;
import io.qt.core.QUntypedPropertyData;

abstract class AbstractMetaObjectUtility {
    static final PropertyFlags ResolveDesignable = null;
    static final PropertyFlags ResolveScriptable = null;
    static final PropertyFlags ResolveEditable = null;
    static final PropertyFlags Editable = null;
    static final PropertyFlags Notify = null;
    static final PropertyFlags ResolveStored = null;
    static final PropertyFlags ResolveUser = null;
    static final PropertyFlags Required = PropertyFlags.Required;
    static final PropertyFlags Bindable = PropertyFlags.Bindable;
    static final PropertyFlags Alias = PropertyFlags.Alias;
    static final MetaObjectFlag PropertyAccessInStaticMetaCall = MetaObjectFlag.PropertyAccessInStaticMetaCall;
    
    static MetaObjectUtility.AnnotationInfo analyzeBindableAnnotation(Method method) {
    	QtPropertyBindable bindable = method.getAnnotation(QtPropertyBindable.class);
        return bindable == null ? null : new MetaObjectUtility.AnnotationInfo(bindable.name(), bindable.enabled());
    }
    
    static boolean isValidBindable(Method method) {
    	return method.getParameterCount()==0 && QUntypedBindable.class.isAssignableFrom(method.getReturnType());
    }
    
    static boolean isValidQProperty(Field field) {
    	return QObject.class==field.getType().getEnclosingClass()
    			&& QUntypedPropertyData.class.isAssignableFrom(field.getType());
    }
    
    static MetaObjectUtility.QPropertyTypeInfo getQPropertyTypeInfo(Field field) {
    	if(field.getType()==QObject.QBooleanProperty.class
    			|| field.getType()==QObject.QComputedBooleanProperty.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(boolean.class, boolean.class, null, false, false, field.getType()==QObject.QBooleanProperty.class);
    	}else if(field.getType()==QObject.QByteProperty.class
    			|| field.getType()==QObject.QComputedByteProperty.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(byte.class, byte.class, null, false, false, field.getType()==QObject.QByteProperty.class);
    	}else if(field.getType()==QObject.QIntProperty.class
    			|| field.getType()==QObject.QComputedIntProperty.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(int.class, int.class, null, false, false, field.getType()==QObject.QIntProperty.class);
    	}else if(field.getType()==QObject.QShortProperty.class
    			|| field.getType()==QObject.QComputedShortProperty.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(short.class, short.class, null, false, false, field.getType()==QObject.QShortProperty.class);
    	}else if(field.getType()==QObject.QLongProperty.class
    			|| field.getType()==QObject.QComputedLongProperty.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(long.class, long.class, null, false, false, field.getType()==QObject.QLongProperty.class);
    	}else if(field.getType()==QObject.QFloatProperty.class
    			|| field.getType()==QObject.QComputedFloatProperty.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(float.class, float.class, null, false, false, field.getType()==QObject.QFloatProperty.class);
    	}else if(field.getType()==QObject.QDoubleProperty.class
    			|| field.getType()==QObject.QComputedDoubleProperty.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(double.class, double.class, null, false, false, field.getType()==QObject.QDoubleProperty.class);
    	}else if(field.getType()==QObject.QCharProperty.class
    			|| field.getType()==QObject.QComputedCharProperty.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(char.class, char.class, null, false, false, field.getType()==QObject.QCharProperty.class);
    	}else if(field.getType()==QObject.QProperty.class
    			|| field.getType()==QObject.QComputedProperty.class) {
    		if(ClassAnalyzerUtility.useAnnotatedType) {
	        	AnnotatedElement t = field.getAnnotatedType();
	        	if (t instanceof AnnotatedParameterizedType) {
	            	AnnotatedParameterizedType p = (AnnotatedParameterizedType) t;
	            	AnnotatedElement actualTypes[] = p.getAnnotatedActualTypeArguments();
	            	if(actualTypes.length==1) {
	            		AnnotatedElement actualType = actualTypes[0];
	                	boolean isPrimitive = actualType.isAnnotationPresent(QtPrimitiveType.class);
	                	boolean isPointer = actualType.isAnnotationPresent(QtPointerType.class);
	                	QtReferenceType referenceType = actualType.getAnnotation(QtReferenceType.class);
	                	boolean isReference = !isPointer && referenceType!=null && !referenceType.isConst();
	                	if(!isPrimitive) {
	                		AnnotatedElement annotatedOwnerType = RetroHelper.getAnnotatedOwnerType(actualType);
	                		if(annotatedOwnerType!=null) {
	                			isPrimitive = annotatedOwnerType.isAnnotationPresent(QtPrimitiveType.class);
	                		}
	                	}
	                	Type type = ((java.lang.reflect.AnnotatedType)actualType).getType();
	                    Class rawType;
	                    if (type instanceof Class) {
	                        rawType = (Class) type;
	                    }else if (type instanceof ParameterizedType) {
	                    	ParameterizedType ptype = (ParameterizedType)type;
	                        rawType = (Class)ptype.getRawType();
	                    } else {
	                    	return null;
	                    }
	                    if(isPrimitive) {
	                    	if(rawType==Integer.class) {
	                    		rawType = int.class;
	                    		type = int.class;
	                    	}else if(rawType==Short.class) {
	                    		rawType = short.class;
	                    		type = short.class;
	                    	}else if(rawType==Byte.class) {
	                    		rawType = byte.class;
	                    		type = byte.class;
	                    	}else if(rawType==Long.class) {
	                    		rawType = long.class;
	                    		type = long.class;
	                    	}else if(rawType==Double.class) {
	                    		rawType = double.class;
	                    		type = double.class;
	                    	}else if(rawType==Float.class) {
	                    		rawType = float.class;
	                    		type = float.class;
	                    	}else if(rawType==Boolean.class) {
	                    		rawType = boolean.class;
	                    		type = boolean.class;
	                    	}else if(rawType==Character.class) {
	                    		rawType = char.class;
	                    		type = char.class;
	                    	}
	                    }
	                    return new MetaObjectUtility.QPropertyTypeInfo(rawType, type, actualType, isPointer, isReference, field.getType()==QObject.QProperty.class);
	            	}
	        	}
    		}else {
    			Type t = field.getGenericType();
            	if (t instanceof ParameterizedType) {
                	ParameterizedType p = (ParameterizedType) t;
                	Type actualTypes[] = p.getActualTypeArguments();
                	if(actualTypes.length==1) {
                		Type type = actualTypes[0];
                        Class rawType;
                        if (type instanceof Class) {
                            rawType = (Class) type;
                        }else if (type instanceof ParameterizedType) {
                        	ParameterizedType ptype = (ParameterizedType)type;
                            rawType = (Class)ptype.getRawType();
                        } else {
                        	return null;
                        }
                        if(rawType==Integer.class) {
                    		rawType = int.class;
                    		type = int.class;
                    	}else if(rawType==Short.class) {
                    		rawType = short.class;
                    		type = short.class;
                    	}else if(rawType==Byte.class) {
                    		rawType = byte.class;
                    		type = byte.class;
                    	}else if(rawType==Long.class) {
                    		rawType = long.class;
                    		type = long.class;
                    	}else if(rawType==Double.class) {
                    		rawType = double.class;
                    		type = double.class;
                    	}else if(rawType==Float.class) {
                    		rawType = float.class;
                    		type = float.class;
                    	}else if(rawType==Boolean.class) {
                    		rawType = boolean.class;
                    		type = boolean.class;
                    	}else if(rawType==Character.class) {
                    		rawType = char.class;
                    		type = char.class;
                    	}
                        return new MetaObjectUtility.QPropertyTypeInfo(rawType, type, null, false, false, field.getType()==QObject.QProperty.class);
                	}
            	}
    		}
    	}
    	return null;
    }
    
    static MetaObjectUtility.QPropertyTypeInfo getQPropertyTypeInfo(Method method) {
    	if(method.getReturnType()==QBooleanBindable.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(boolean.class, boolean.class, null, false, false, method.getReturnType()==QBooleanBindable.class);
    	}else if(method.getReturnType()==QByteBindable.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(byte.class, byte.class, null, false, false, method.getReturnType()==QByteBindable.class);
    	}else if(method.getReturnType()==QIntBindable.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(int.class, int.class, null, false, false, method.getReturnType()==QIntBindable.class);
    	}else if(method.getReturnType()==QShortBindable.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(short.class, short.class, null, false, false, method.getReturnType()==QShortBindable.class);
    	}else if(method.getReturnType()==QLongBindable.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(long.class, long.class, null, false, false, method.getReturnType()==QLongBindable.class);
    	}else if(method.getReturnType()==QFloatBindable.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(float.class, float.class, null, false, false, method.getReturnType()==QFloatBindable.class);
    	}else if(method.getReturnType()==QDoubleBindable.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(double.class, double.class, null, false, false, method.getReturnType()==QDoubleBindable.class);
    	}else if(method.getReturnType()==QCharBindable.class) {
    		return new MetaObjectUtility.QPropertyTypeInfo(char.class, char.class, null, false, false, method.getReturnType()==QCharBindable.class);
    	}else if(method.getReturnType()==QBindable.class) {
    		if(ClassAnalyzerUtility.useAnnotatedType) {
	        	AnnotatedElement t = method.getAnnotatedReturnType();
	        	if (t instanceof AnnotatedParameterizedType) {
	            	AnnotatedParameterizedType p = (AnnotatedParameterizedType) t;
	            	AnnotatedElement actualTypes[] = p.getAnnotatedActualTypeArguments();
	            	if(actualTypes.length==1) {
	            		AnnotatedElement actualType = actualTypes[0];
	                	boolean isPrimitive = actualType.isAnnotationPresent(QtPrimitiveType.class);
	                	boolean isPointer = actualType.isAnnotationPresent(QtPointerType.class);
	                	QtReferenceType referenceType = actualType.getAnnotation(QtReferenceType.class);
	                	boolean isReference = !isPointer && referenceType!=null && !referenceType.isConst();
	                	if(!isPrimitive) {
	                		AnnotatedElement annotatedOwnerType = RetroHelper.getAnnotatedOwnerType(actualType);
	                		if(annotatedOwnerType!=null) {
	                			isPrimitive = annotatedOwnerType.isAnnotationPresent(QtPrimitiveType.class);
	                		}
	                	}
	                	Type type = ((java.lang.reflect.AnnotatedType)actualType).getType();
	                    Class rawType;
	                    if (type instanceof Class) {
	                        rawType = (Class) type;
	                    }else if (type instanceof ParameterizedType) {
	                    	ParameterizedType ptype = (ParameterizedType)type;
	                        rawType = (Class)ptype.getRawType();
	                    } else {
	                    	return null;
	                    }
	                    if(isPrimitive) {
	                    	if(rawType==Integer.class) {
	                    		rawType = int.class;
	                    		type = int.class;
	                    	}else if(rawType==Short.class) {
	                    		rawType = short.class;
	                    		type = short.class;
	                    	}else if(rawType==Byte.class) {
	                    		rawType = byte.class;
	                    		type = byte.class;
	                    	}else if(rawType==Long.class) {
	                    		rawType = long.class;
	                    		type = long.class;
	                    	}else if(rawType==Double.class) {
	                    		rawType = double.class;
	                    		type = double.class;
	                    	}else if(rawType==Float.class) {
	                    		rawType = float.class;
	                    		type = float.class;
	                    	}else if(rawType==Boolean.class) {
	                    		rawType = boolean.class;
	                    		type = boolean.class;
	                    	}else if(rawType==Character.class) {
	                    		rawType = char.class;
	                    		type = char.class;
	                    	}
	                    }
	                    return new MetaObjectUtility.QPropertyTypeInfo(rawType, type, actualType, isPointer, isReference, method.getReturnType()==QBindable.class);
	            	}
	        	}
    		}else {
    			Type t = method.getGenericReturnType();
            	if (t instanceof ParameterizedType) {
                	ParameterizedType p = (ParameterizedType) t;
                	Type actualTypes[] = p.getActualTypeArguments();
                	if(actualTypes.length==1) {
                		Type type = actualTypes[0];
                        Class rawType;
                        if (type instanceof Class) {
                            rawType = (Class) type;
                        }else if (type instanceof ParameterizedType) {
                        	ParameterizedType ptype = (ParameterizedType)type;
                            rawType = (Class)ptype.getRawType();
                        } else {
                        	return null;
                        }
                        if(rawType==Integer.class) {
                    		rawType = int.class;
                    		type = int.class;
                    	}else if(rawType==Short.class) {
                    		rawType = short.class;
                    		type = short.class;
                    	}else if(rawType==Byte.class) {
                    		rawType = byte.class;
                    		type = byte.class;
                    	}else if(rawType==Long.class) {
                    		rawType = long.class;
                    		type = long.class;
                    	}else if(rawType==Double.class) {
                    		rawType = double.class;
                    		type = double.class;
                    	}else if(rawType==Float.class) {
                    		rawType = float.class;
                    		type = float.class;
                    	}else if(rawType==Boolean.class) {
                    		rawType = boolean.class;
                    		type = boolean.class;
                    	}else if(rawType==Character.class) {
                    		rawType = char.class;
                    		type = char.class;
                    	}
                        return new MetaObjectUtility.QPropertyTypeInfo(rawType, type, null, false, false, method.getReturnType()==QBindable.class);
                	}
            	}
    		}
    	}
    	return null;
    }
    
    private static class PropertyIndex{
    	private PropertyIndex(Field field, int propertyIndex) {
			super();
			this.field = field;
			this.propertyIndex = propertyIndex;
		}
		final Field field;
		final int propertyIndex;
    }
    
    private final static Map, List> propertyFieldsByClasses = Collections.synchronizedMap(new HashMap<>());
    
    static  PI analyzeProperty(QObject containingObject, QtObject property, BiFunction fun1, BiFunction fun2) {
		List propertyFields = propertyFieldsByClasses.computeIfAbsent(AccessUtility.instance.getClass(containingObject), cls->{
			List fields = Collections.emptyList();
			while (QObject.class.isAssignableFrom(cls)) {
				QMetaObject metaObject = QMetaObject.forType(cls);
	            for (Field field : cls.getDeclaredFields()) {
	                if (!Modifier.isStatic(field.getModifiers())
            			&& AbstractMetaObjectUtility.isValidQProperty(field)) {
	                	QMetaProperty metaProperty = getPropertyForField(metaObject, field);
	                	PropertyIndex info = new PropertyIndex(field, metaProperty!=null ? metaProperty.propertyIndex() : -1);
						if(fields.isEmpty()) {
	        				fields = Collections.singletonList(info);
	        			} else {
	        				if(fields.size()==1)
    	        				fields = new ArrayList<>(fields);
	        				fields.add(info);
	        			}
	                }
	            }
	            cls = cls.getSuperclass();
	        }
			if(fields.isEmpty()) {
				return Collections.emptyList();
			}else if(fields.size()==1){
				return fields;
			}else {
				return Collections.synchronizedList(fields);
			}
		});
		
		Field foundField = null;
		QList remainingProperties = containingObject.metaObject().properties();
		for (PropertyIndex info : propertyFields) {
			try {
				if(ReflectionUtility.readField(containingObject, info.field)==property) {
					if(info.propertyIndex>=0) {
						QMetaProperty metaProperty = containingObject.metaObject().properties().at(info.propertyIndex);
						remainingProperties.removeOne(metaProperty);
            			if(!Modifier.isFinal(info.field.getModifiers())) {
            				if(!Boolean.getBoolean("qtjambi.allow-nonfinal-qproperties") && !Boolean.getBoolean("io.qt.allow-nonfinal-qproperties")) {
                    			java.util.logging.Logger.getLogger("io.qt.internal").severe(String.format("Missing modifier 'final' at property field %1$s.%2$s. Specify JVM argument -Dqtjambi.allow-nonfinal-qproperties=true to disable this error.", info.field.getDeclaringClass().getSimpleName(), info.field.getName()));
                    			throw new QPropertyDeclarationException(String.format("Missing modifier 'final' at property field %1$s.%2$s.", info.field.getDeclaringClass().getSimpleName(), info.field.getName()));
                    		}
            			}
	    				return fun2.apply(info.field, metaProperty);
					}else {
						foundField = info.field;
					}
					break;
				}
			} catch (QPropertyDeclarationException e) {
				throw e;
			} catch (Throwable e) {}
        }
		if(foundField==null)
			throw new QPropertyDeclarationException("Cannot find member field belonging to QProperty instance.");
		MetaObjectUtility.QPropertyTypeInfo pinfo = getQPropertyTypeInfo(foundField);
		int t = MetaTypeUtility.registerMetaType(
				pinfo.propertyType, 
				pinfo.genericPropertyType, 
				pinfo.annotatedPropertyType,
				pinfo.isPointer,
				pinfo.isReference);
		return fun1.apply(foundField, new QMetaType(t));
    }
    
    static void registerPropertyField(QMetaProperty metaProperty, java.lang.reflect.Field field) {
    	registerPropertyField(NativeUtility.nativeId(metaProperty), field);
    }
    
    private static native void registerPropertyField(long metaPropertyId, java.lang.reflect.Field field);
    
    private static native @Nullable QMetaProperty getPropertyForField(QMetaObject metaObject, Field field);
    
    static boolean isListType(Class cls) {
    	return cls==QList.class
    			|| cls==QQueue.class
    			|| cls==QStack.class
    			|| cls==QSet.class;
    }
    
    static boolean isSequentialContainer(Class cls) {
    	return QList.class.isAssignableFrom(cls)
    			|| QSet.class.isAssignableFrom(cls);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy