
io.qt.internal.QtJambiEnums 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) 1992-2009 Nokia. All rights reserved.
** 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 java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodType;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.function.Function;
import io.qt.NativeAccess;
import io.qt.QFlags;
import io.qt.QtAbstractFlagEnumerator;
import io.qt.QtByteFlagEnumerator;
import io.qt.QtExtensibleEnum;
import io.qt.QtFlagEnumerator;
import io.qt.QtLongFlagEnumerator;
import io.qt.QtShortFlagEnumerator;
public final class QtJambiEnums {
private QtJambiEnums() {throw new RuntimeException();}
private static final Map, MethodHandle> flagsConstructorsByEnumType;
private static final Map, QtAbstractFlagEnumerator[]> flagsConstants;
static {
flagsConstructorsByEnumType = Collections.synchronizedMap(new HashMap<>());
flagsConstants = Collections.synchronizedMap(new HashMap<>());
}
public static QFlags> asFlags(QtAbstractFlagEnumerator flag, Function> constructor) {
MethodHandle flagsConstructor = flagsConstructorsByEnumType.computeIfAbsent(flag.getClass(), cls -> {
Class> declClass = cls.getDeclaringClass();
if (declClass != null) {
for (Class> flagClass : declClass.getDeclaredClasses()) {
if (flagClass.getSuperclass() == QFlags.class) {
if (flagClass.getGenericSuperclass() instanceof ParameterizedType) {
ParameterizedType p = (ParameterizedType) flagClass.getGenericSuperclass();
Type[] args = p.getActualTypeArguments();
if (args.length == 1 && args[0] == cls) {
try {
Constructor> constr = flagClass.getDeclaredConstructor(int.class);
return QtJambiInternal.privateLookup(flagClass).unreflectConstructor(constr);
} catch (Throwable e) {
}
try {
Constructor> constr = flagClass.getDeclaredConstructor();
return QtJambiInternal.privateLookup(flagClass).unreflectConstructor(constr);
} catch (Throwable e) {
}
try {
Constructor> constr = flagClass.getConstructor();
return QtJambiInternal.privateLookup(flagClass).unreflectConstructor(constr);
} catch (Throwable e) {
}
}
}
}
}
}
return null;
});
int value = 0;
if (flag instanceof QtFlagEnumerator) {
value |= ((QtFlagEnumerator) flag).value();
} else if (flag instanceof QtByteFlagEnumerator) {
value |= ((QtByteFlagEnumerator) flag).value();
} else if (flag instanceof QtShortFlagEnumerator) {
value |= ((QtShortFlagEnumerator) flag).value();
} else if (flag instanceof QtLongFlagEnumerator) {
value |= ((QtLongFlagEnumerator) flag).value();
}
if (flagsConstructor != null) {
if (flagsConstructor.type().parameterCount() == 1) {
try {
return (QFlags>) flagsConstructor.invoke(value);
} catch (Throwable e) {
}
} else {
try {
QFlags> flags = (QFlags>) flagsConstructor.invoke();
flags.setValue(flags.value() | value);
return flags;
} catch (Throwable e) {
}
}
}
return constructor.apply(value);
}
public static boolean isSmallEnum(QtAbstractFlagEnumerator enm) {
try {
QtAbstractFlagEnumerator[] values;
if (enm.getClass().isAnnotationPresent(QtExtensibleEnum.class)) {
values = enm.getClass().getEnumConstants();
} else {
values = flagsConstants.computeIfAbsent(enm.getClass(), cls -> {
return cls.getEnumConstants();
});
}
return values != null && values.length <= 33;
} catch (Exception e) {
}
return false;
}
@SuppressWarnings("unchecked")
public static T[] flagConstants(QFlags flags) {
if (flags.getClass().getGenericSuperclass() instanceof ParameterizedType) {
ParameterizedType superType = (ParameterizedType) flags.getClass().getGenericSuperclass();
if (superType.getRawType() == QFlags.class) {
Type[] typeArguments = superType.getActualTypeArguments();
if (typeArguments.length == 1 && typeArguments[0] instanceof Class) {
Class enumFlagType = (Class) typeArguments[0];
T[] values;
if (enumFlagType.isAnnotationPresent(QtExtensibleEnum.class)) {
values = enumFlagType.getEnumConstants();
} else {
values = (T[]) flagsConstants.computeIfAbsent(enumFlagType, cls -> {
return cls.getEnumConstants();
});
}
return values;
}
}
}
return (T[]) new QtFlagEnumerator[0];
}
@NativeAccess
private static > boolean extendEnum(Class enumClass, T[] array, T enumEntry) {
java.lang.invoke.MethodHandles.Lookup pl = QtJambiInternal.privateLookup(Class.class);
try {
Field enumConstantsField = Class.class.getDeclaredField("enumConstants");
Field enumConstantDirectoryField = Class.class.getDeclaredField("enumConstantDirectory");
pl.unreflectSetter(enumConstantsField).invoke(enumClass, array);
Map directory = (Map) pl.unreflectGetter(enumConstantDirectoryField)
.invoke(enumClass);
if (directory != null) {
directory.put(enumEntry.name(), enumEntry);
}
return true;
} catch (Throwable e) {
try {
Field enumVarsField = Class.class.getDeclaredField("enumVars");
Object enumVars = pl.unreflectGetter(enumVarsField).invoke(enumClass);
java.lang.invoke.MethodHandles.Lookup pl2 = QtJambiInternal.privateLookup(enumVarsField.getType());
if (enumVars == null) {
enumVars = pl2.findConstructor(enumVarsField.getType(), MethodType.methodType(void.class)).invoke();
pl.unreflectSetter(enumVarsField).invoke(enumClass, enumVars);
}
Field enumConstantsField = enumVarsField.getType().getDeclaredField("cachedEnumConstants");
Field enumConstantDirectoryField = enumVarsField.getType()
.getDeclaredField("cachedEnumConstantDirectory");
pl2.unreflectSetter(enumConstantsField).invoke(enumVars, array);
Map directory = (Map) pl2.unreflectGetter(enumConstantDirectoryField)
.invoke(enumVars);
if (directory != null) {
directory.put(enumEntry.name(), enumEntry);
}
return true;
} catch (Throwable e1) {
e1.printStackTrace();
}
}
return false;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy