io.qt.internal.EnumUtility 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-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.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 java.util.function.Supplier;
import io.qt.NativeAccess;
import io.qt.QFlags;
import io.qt.QNoSuchEnumValueException;
import io.qt.QtAbstractFlagEnumerator;
import io.qt.QtByteEnumerator;
import io.qt.QtByteFlagEnumerator;
import io.qt.QtEnumerator;
import io.qt.QtExtensibleEnum;
import io.qt.QtFlagEnumerator;
import io.qt.QtLongEnumerator;
import io.qt.QtLongFlagEnumerator;
import io.qt.QtShortEnumerator;
import io.qt.QtShortFlagEnumerator;
/**
* @hidden
*/
public final class EnumUtility {
static {
QtJambi_LibraryUtilities.initialize();
}
private EnumUtility() {throw new RuntimeException();}
private static final Map, Object> flagsConstructorsByEnumType;
private static final Map, QtAbstractFlagEnumerator[]> flagsConstants;
static {
flagsConstructorsByEnumType = Collections.synchronizedMap(new HashMap<>());
flagsConstants = Collections.synchronizedMap(new HashMap<>());
}
@SuppressWarnings("unchecked")
public static QFlags> asFlags(QtAbstractFlagEnumerator flag, Function> constructor) {
Object flagsConstructor = flagsConstructorsByEnumType.computeIfAbsent(ClassAnalyzerUtility.getClass(flag), 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) {
Constructor> constr = null;
try {
constr = flagClass.getDeclaredConstructor(int.class);
} catch (Throwable e) {
}
if(constr==null) try {
constr = flagClass.getDeclaredConstructor();
} catch (Throwable e) {
}
if(constr==null) try {
constr = flagClass.getConstructor();
} catch (Throwable e) {
}
if(constr!=null) {
if(constr.getParameterCount()==0) {
return ReflectionUtility.getFactory0(constr);
}else {
return ReflectionUtility.getFactory1(constr);
}
}
}
}
}
}
}
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 instanceof Function) {
try {
return ((Function>)flagsConstructor).apply(value);
} catch (Throwable e) {
}
} else if (flagsConstructor instanceof Supplier) {
try {
QFlags> flags = ((Supplier>) flagsConstructor).get();
flags.setValue(flags.value() | value);
return flags;
} catch (Throwable e) {
}
}
}
return constructor.apply(value);
}
public static boolean isSmallEnum(QtAbstractFlagEnumerator enm) {
try {
QtAbstractFlagEnumerator[] values;
Class extends QtAbstractFlagEnumerator> cls = ClassAnalyzerUtility.getClass(enm);
if (cls.isAnnotationPresent(QtExtensibleEnum.class)) {
values = cls.getEnumConstants();
} else {
values = flagsConstants.computeIfAbsent(cls, _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) {
try {
Field enumConstantsField = Class.class.getDeclaredField("enumConstants");
Field enumConstantDirectoryField = Class.class.getDeclaredField("enumConstantDirectory");
ReflectionUtility.writeField(enumClass, enumConstantsField, array);
@SuppressWarnings("unchecked")
Map directory = (Map)ReflectionUtility.readField(enumClass, enumConstantDirectoryField);
if (directory != null) {
directory.put(enumEntry.name(), enumEntry);
}
return true;
} catch (Throwable e) {
try {
Field enumVarsField = Class.class.getDeclaredField("enumVars");
Object enumVars = ReflectionUtility.readField(enumClass, enumVarsField);
if (enumVars == null) {
Constructor> enumVarsConstr = enumVarsField.getType().getDeclaredConstructor();
enumVars = ReflectionUtility.invokeContructor(enumVarsConstr);
ReflectionUtility.writeField(enumClass, enumVarsField, enumVars);
}
Field enumConstantsField = enumVarsField.getType().getDeclaredField("cachedEnumConstants");
Field enumConstantDirectoryField = enumVarsField.getType().getDeclaredField("cachedEnumConstantDirectory");
ReflectionUtility.writeField(enumVars, enumConstantsField, array);
@SuppressWarnings("unchecked")
Map directory = (Map) ReflectionUtility.readField(enumVars, enumConstantDirectoryField);
if (directory != null) {
directory.put(enumEntry.name(), enumEntry);
}
return true;
} catch (Throwable e1) {
if(!LibraryUtility.operatingSystem.isAndroid()) {
e.addSuppressed(e1);
e.printStackTrace();
}
}
}
return false;
}
@NativeAccess
static Class> getEnumForQFlags(Class> flagsType) {
Type t = flagsType.getGenericSuperclass();
if (t instanceof ParameterizedType) {
Type typeArguments[] = ((ParameterizedType)t).getActualTypeArguments();
return ((Class>) typeArguments[0]);
}
return null;
}
private native static & QtEnumerator> E resolveIntEnum(int hashCode, Class cl, int value, String name) throws Throwable;
private native static & QtByteEnumerator> E resolveByteEnum(int hashCode, Class cl, byte value, String name) throws Throwable;
private native static & QtShortEnumerator> E resolveShortEnum(int hashCode, Class cl, short value, String name) throws Throwable;
private native static & QtLongEnumerator> E resolveLongEnum(int hashCode, Class cl, long value, String name) throws Throwable;
static & QtEnumerator> E resolveEnum(Class cl, int value, String name) {
if (name != null) {
if (name.isEmpty())
name = null;
else {
E enm = null;
try {
enm = Enum.valueOf(cl, name);
} catch (Exception e) {
}
if (enm != null) {
if (enm.value() == value) {
return enm;
} else {
throw new io.qt.QNoSuchEnumValueException(value, name);
}
}
}
}
try {
E enm = EnumUtility.resolveIntEnum(cl.hashCode(), cl, value, name);
if (enm == null) {
if (name == null)
throw new QNoSuchEnumValueException(value);
else
throw new QNoSuchEnumValueException(value, name);
}
return enm;
} catch (QNoSuchEnumValueException e) {
throw e;
} catch (Throwable e) {
throw new QNoSuchEnumValueException(value, e);
}
}
static & QtByteEnumerator> E resolveEnum(Class cl, byte value, String name) {
if (name != null) {
if (name.isEmpty())
name = null;
else {
E enm = null;
try {
enm = Enum.valueOf(cl, name);
} catch (Exception e) {
}
if (enm != null) {
if (enm.value() == value) {
return enm;
} else {
throw new io.qt.QNoSuchEnumValueException(value, name);
}
}
}
}
try {
E enm = EnumUtility.resolveByteEnum(cl.hashCode(), cl, value, name);
if (enm == null) {
if (name == null)
throw new QNoSuchEnumValueException(value);
else
throw new QNoSuchEnumValueException(value, name);
}
return enm;
} catch (QNoSuchEnumValueException e) {
throw e;
} catch (Throwable e) {
throw new QNoSuchEnumValueException(value, e);
}
}
static & QtShortEnumerator> E resolveEnum(Class cl, short value, String name) {
if (name != null) {
if (name.isEmpty())
name = null;
else {
E enm = null;
try {
enm = Enum.valueOf(cl, name);
} catch (Exception e) {
}
if (enm != null) {
if (enm.value() == value) {
return enm;
} else {
throw new io.qt.QNoSuchEnumValueException(value, name);
}
}
}
}
try {
E enm = EnumUtility.resolveShortEnum(cl.hashCode(), cl, value, name);
if (enm == null) {
if (name == null)
throw new QNoSuchEnumValueException(value);
else
throw new QNoSuchEnumValueException(value, name);
}
return enm;
} catch (QNoSuchEnumValueException e) {
throw e;
} catch (Throwable e) {
throw new QNoSuchEnumValueException(value, e);
}
}
static & QtLongEnumerator> E resolveEnum(Class cl, long value, String name) {
if (name != null) {
if (name.isEmpty())
name = null;
else {
E enm = null;
try {
enm = Enum.valueOf(cl, name);
} catch (Exception e) {
}
if (enm != null) {
if (enm.value() == value) {
return enm;
} else {
throw new io.qt.QNoSuchEnumValueException(value, name);
}
}
}
}
try {
E enm = EnumUtility.resolveLongEnum(cl.hashCode(), cl, value, name);
if (enm == null) {
if (name == null)
throw new QNoSuchEnumValueException(value);
else
throw new QNoSuchEnumValueException(value, name);
}
return enm;
} catch (QNoSuchEnumValueException e) {
throw e;
} catch (Throwable e) {
throw new QNoSuchEnumValueException(value, e);
}
}
}