io.qt.internal.AbstractMetaObjectTools Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of qtjambi Show documentation
Show all versions of qtjambi Show documentation
QtJambi base module containing QtCore, QtGui and QtWidgets.
/****************************************************************************
**
** Copyright (C) 2009-2022 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 static io.qt.internal.QtJambiInternal.registerMetaType;
import java.lang.invoke.MethodHandle;
import java.lang.reflect.AnnotatedParameterizedType;
import java.lang.reflect.AnnotatedType;
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 io.qt.QtObject;
import io.qt.QtPointerType;
import io.qt.QtPrimitiveType;
import io.qt.QtPropertyBindable;
import io.qt.QtReferenceType;
import io.qt.core.QList;
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.QStack;
import io.qt.core.QUntypedBindable;
import io.qt.core.QUntypedPropertyData;
class AbstractMetaObjectTools {
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 MetaObjectTools.AnnotationInfo analyzeBindableAnnotation(Method method) {
QtPropertyBindable bindable = method.getAnnotation(QtPropertyBindable.class);
return bindable == null ? null : new MetaObjectTools.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 MetaObjectTools.QPropertyTypeInfo getQPropertyTypeInfo(Field field) {
if(field.getType()==QObject.QBooleanProperty.class
|| field.getType()==QObject.QComputedBooleanProperty.class) {
return new MetaObjectTools.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 MetaObjectTools.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 MetaObjectTools.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 MetaObjectTools.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 MetaObjectTools.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 MetaObjectTools.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 MetaObjectTools.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 MetaObjectTools.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) {
AnnotatedType t = field.getAnnotatedType();
if (t instanceof AnnotatedParameterizedType) {
AnnotatedParameterizedType p = (AnnotatedParameterizedType) t;
AnnotatedType actualTypes[] = p.getAnnotatedActualTypeArguments();
if(actualTypes.length==1) {
AnnotatedType 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) {
AnnotatedType annotatedOwnerType = RetroHelper.getAnnotatedOwnerType(actualType);
if(annotatedOwnerType!=null) {
isPrimitive = annotatedOwnerType.isAnnotationPresent(QtPrimitiveType.class);
}
}
Type type = 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 MetaObjectTools.QPropertyTypeInfo(rawType, type, actualType, isPointer, isReference, field.getType()==QObject.QProperty.class);
}
}
}
return null;
}
private static class PropertyInfo{
private PropertyInfo(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 QtJambiPropertyInfo analyzeProperty(QObject containingObject, QtObject property) {
List propertyFields = propertyFieldsByClasses.computeIfAbsent(containingObject.getClass(), 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())
&& AbstractMetaObjectTools.isValidQProperty(field)) {
QMetaProperty metaProperty = getPropertyForField(metaObject, field);
PropertyInfo info = new PropertyInfo(field, metaProperty.isValid() ? 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 (PropertyInfo info : propertyFields) {
try {
MethodHandle fieldGetter = QtJambiInternal.getFieldGetterHandle(info.field);
if(fieldGetter.invoke(containingObject)==property) {
if(info.propertyIndex>=0) {
QMetaProperty metaProperty = containingObject.metaObject().properties().at(info.propertyIndex);
remainingProperties.removeOne(metaProperty);
if(!Modifier.isFinal(info.field.getModifiers()))
throw new RuntimeException("QProperty field "+info.field.getName()+" is not final.");
return new QtJambiPropertyInfo(info.field, metaProperty);
}else {
foundField = info.field;
}
break;
}
} catch (Throwable e) {}
}
if(foundField==null)
throw new RuntimeException("Cannot find member field belonging to QProperty instance.");
MetaObjectTools.QPropertyTypeInfo pinfo = getQPropertyTypeInfo(foundField);
int t = registerMetaType(
pinfo.propertyType,
pinfo.genericPropertyType,
pinfo.annotatedPropertyType,
pinfo.isPointer,
pinfo.isReference);
return new QtJambiPropertyInfo(foundField, new QMetaType(t));
}
static void registerPropertyField(QMetaProperty metaProperty, java.lang.reflect.Field field) {
registerPropertyField(QtJambiInternal.internalAccess.nativeId(metaProperty), field);
}
private static native void registerPropertyField(long metaPropertyId, java.lang.reflect.Field field);
private static native QMetaProperty getPropertyForField(QMetaObject metaObject, Field field);
static boolean isListType(Class> cls) {
return cls==QList.class
|| cls==QQueue.class
|| cls==QStack.class
|| cls==QSet.class;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy