
io.qt.internal.EnumUtility 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.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);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy