io.qt.core.QScope 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.core;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.function.Consumer;
import java.util.logging.Level;
import java.util.logging.Logger;
import io.qt.*;
import io.qt.QtObjectInterface;
/**
* QScope is an object life-time manager to be used in a try-with-resource block.
* All object created with this scope are cleaned up (disposed) at the end of the scope.
* Example:
*
* try(QScope scope = new QScope()){
* QDialog dialog = scope.create(QDialog::new);
* dialog.exec();
* }
* // dialog is disposed
*
*/
public final class QScope implements AutoCloseable {
private static final Logger logger = Logger.getLogger("io.qt");
static abstract class AbstractEntry{
AbstractEntry(O data) {
this.data = data;
}
O data;
abstract void cleanup();
void dismiss() {
data = null;
}
}
static class RunningEntry extends AbstractEntry{
RunningEntry(Runnable data) {
super(data);
}
void cleanup(){
if(data!=null) {
try {
data.run();
} catch (Throwable e) {
logger.log(Level.WARNING, "scope cleanup", e);
}
}
data = null;
}
}
static class DisposingEntry extends AbstractEntry{
DisposingEntry(QtObjectInterface data) {
super(data);
}
void cleanup(){
if(data!=null) {
try {
data.dispose();
} catch (Throwable e) {
logger.log(Level.WARNING, "scope cleanup", e);
}
}
data = null;
}
}
static class DisposingLaterEntry extends AbstractEntry{
DisposingLaterEntry(QObject data) {
super(data);
}
void cleanup(){
if(data!=null) {
try {
data.disposeLater();
} catch (Throwable e) {
logger.log(Level.WARNING, "scope cleanup", e);
}
}
data = null;
}
}
static class CleanupEntry extends AbstractEntry{
CleanupEntry(O data, Consumer cleanup) {
super(data);
this.cleanup = Objects.requireNonNull(cleanup);
}
private Consumer cleanup;
void cleanup(){
if(cleanup!=null) {
try{
if(data!=null) {
try {
cleanup.accept(data);
} catch (Throwable e) {
logger.log(Level.WARNING, "scope cleanup", e);
}
}
}finally{
cleanup = null;
}
}
data = null;
}
void dismiss() {
super.dismiss();
cleanup = null;
}
Consumer getCleanup() {
return cleanup;
}
}
private static class Data{
private final List> objects = new ArrayList<>();
private InternalAccess.Cleanable cleanable;
@QtUninvokable
void close(){
for(AbstractEntry> e : objects) {
e.cleanup();
}
objects.clear();
}
void addEntry(AbstractEntry> entry) {
if(this.cleanable==null)
throw new IllegalStateException("Scope has already been closed.");
objects.add(entry);
}
}
private final Data data = new Data();
/**
* Default constructor
*/
public QScope() {
super();
this.data.cleanable = QtJambi_LibraryUtilities.internal.registerCleaner(this, this.data::close);
}
/**
* Performs a cleanup on all
*/
@Override
@QtUninvokable
public void close(){
if(this.data.cleanable!=null) {
this.data.cleanable.clean();
this.data.cleanable = null;
}
}
/**
* Removes the given object from this scope.
* @param object
* @return success
*/
@QtUninvokable
public boolean release(Object object) {
if(this.data.cleanable==null)
throw new IllegalStateException("Scope has already been closed.");
for(int i=0; i entry = this.data.objects.get(i);
if(entry.data==object) {
this.data.objects.remove(i);
entry.dismiss();
return true;
}
}
return false;
}
/**
* Forces the object's cleanup.
* @param object
* @return success
*/
@QtUninvokable
public boolean cleanup(Object object) {
if(this.data.cleanable==null)
throw new IllegalStateException("Scope has already been closed.");
for(int i=0; i entry = this.data.objects.get(i);
if(entry.data==object) {
this.data.objects.remove(i);
entry.cleanup();
return true;
}
}
return false;
}
/**
* Let given action be performed at the end of the scope.
* @param action
*/
@QtUninvokable
public Runnable capture(Runnable action) {
data.addEntry(new RunningEntry(action));
return action;
}
/**
* The given object is managed by the scope, i.e. disposed at its closing.
* @param object
* @return object
*/
@QtUninvokable
public T capture(T object){
data.addEntry(new DisposingEntry(object));
return object;
}
/**
* The given object is managed by the scope, i.e. cleaned up at its closing.
* @param cleanup
* @param object
* @return object
*/
@QtUninvokable
public T capture(Consumer cleanup, T object){
data.addEntry(new CleanupEntry<>(object, cleanup));
return object;
}
/**
* Factory for argument-less constructors
* @param the constructed type
*/
@FunctionalInterface
public static interface Factory {
/**
* Creating the object
* @return new object
*/
public R create();
}
/**
* Factory for one argument taking constructors
* @param the constructed type
*/
@FunctionalInterface
public static interface Factory1 {
/**
* Creating the object
* @param a
* @return new object
*/
public R create(A a);
}
/**
* Factory for two arguments taking constructors
* @param the constructed type
* @param type of argument 1
* @param type of argument 2
*/
@FunctionalInterface
public static interface Factory2 {
/**
* Creating the object
* @param a
* @param b
* @return new object
*/
public R create(A a, B b);
}
/**
* Factory for 3 arguments taking constructors
* @param the constructed type
* @param type of argument 1
* @param type of argument 2
* @param type of argument 3
*/
@FunctionalInterface
public static interface Factory3 {
/**
* Creating the object
* @param a
* @param b
* @param c
* @return new object
*/
public R create(A a, B b, C c);
}
/**
* Factory for 4 arguments taking constructors
* @param the constructed type
* @param type of argument 1
* @param type of argument 2
* @param type of argument 3
* @param type of argument 4
*/
@FunctionalInterface
public static interface Factory4{
/**
* Creating the object
* @param a
* @param b
* @param c
* @param d
* @return new object
*/
public R create(A a, B b, C c, D d);
}
/**
* Factory for 5 arguments taking constructors
* @param the constructed type
* @param type of argument 1
* @param type of argument 2
* @param type of argument 3
* @param type of argument 4
* @param type of argument 5
*/
@FunctionalInterface
public static interface Factory5{
/**
* Creating the object
* @param a
* @param b
* @param c
* @param d
* @param e
* @return new object
*/
public R create(A a, B b, C c, D d, E e);
}
/**
* Factory for 6 arguments taking constructors
* @param the constructed type
* @param type of argument 1
* @param type of argument 2
* @param type of argument 3
* @param type of argument 4
* @param type of argument 5
* @param type of argument 6
*/
@FunctionalInterface
public static interface Factory6{
/**
* Creating the object
* @param a
* @param b
* @param c
* @param d
* @param e
* @param f
* @return new object
*/
public R create(A a, B b, C c, D d, E e, F f);
}
/**
* Factory for 7 arguments taking constructors
* @param the constructed type
* @param type of argument 1
* @param type of argument 2
* @param type of argument 3
* @param type of argument 4
* @param type of argument 5
* @param type of argument 6
* @param type of argument 7
*/
@FunctionalInterface
public static interface Factory7{
/**
* Creating the object
* @param a
* @param b
* @param c
* @param d
* @param e
* @param f
* @param g
* @return new object
*/
public R create(A a, B b, C c, D d, E e, F f, G g);
}
/**
* Factory for 8 arguments taking constructors
* @param the constructed type
* @param type of argument 1
* @param type of argument 2
* @param type of argument 3
* @param type of argument 4
* @param type of argument 5
* @param type of argument 6
* @param type of argument 7
* @param type of argument 8
*/
@FunctionalInterface
public static interface Factory8{
/**
* Creating the object
* @param a
* @param b
* @param c
* @param d
* @param e
* @param f
* @param g
* @param h
* @return new object
*/
public R create(A a, B b, C c, D d, E e, F f, G g, H h);
}
/**
* Factory for 9 arguments taking constructors
* @param the constructed type
* @param type of argument 1
* @param type of argument 2
* @param type of argument 3
* @param type of argument 4
* @param type of argument 5
* @param type of argument 6
* @param type of argument 7
* @param type of argument 8
* @param type of argument 9
*/
@FunctionalInterface
public static interface Factory9{
/**
* Creating the object
* @param a
* @param b
* @param c
* @param d
* @param e
* @param f
* @param g
* @param h
* @param i
* @return new object
*/
public R create(A a, B b, C c, D d, E e, F f, G g, H h, I i);
}
/**
* Creates a new object being managed by this scope.
* @param factory constructor
* @return new scoped object
*/
public O create(Factory factory) {
return capture(factory.create());
}
/**
* Creates a new object being managed by this scope.
* @param factory constructor
* @return new scoped object
*/
public O create(Factory1 factory, A arg0) {
return capture(factory.create(arg0));
}
/**
* Creates a new object being managed by this scope.
* @param factory constructor
* @return new scoped object
*/
public O create(Factory2 factory, A arg0, B arg1) {
return capture(factory.create(arg0, arg1));
}
/**
* Creates a new object being managed by this scope.
* @param factory constructor
* @return new scoped object
*/
public O create(Factory3 factory, A arg0, B arg1, C arg2) {
return capture(factory.create(arg0, arg1, arg2));
}
/**
* Creates a new object being managed by this scope.
* @param factory constructor
* @return new scoped object
*/
public O create(Factory4 factory, A arg0, B arg1, C arg2, D arg3) {
return capture(factory.create(arg0, arg1, arg2, arg3));
}
/**
* Creates a new object being managed by this scope.
* @param factory constructor
* @return new scoped object
*/
public O create(Factory5 factory, A arg0, B arg1, C arg2, D arg3, E arg4) {
return capture(factory.create(arg0, arg1, arg2, arg3, arg4));
}
/**
* Creates a new object being managed by this scope.
* @param factory constructor
* @return new scoped object
*/
public O create(Factory6 factory, A arg0, B arg1, C arg2, D arg3, E arg4, F arg5) {
return capture(factory.create(arg0, arg1, arg2, arg3, arg4, arg5));
}
/**
* Creates a new object being managed by this scope.
* @param factory constructor
* @return new scoped object
*/
public O create(Factory7 factory, A arg0, B arg1, C arg2, D arg3, E arg4, F arg5, G arg6) {
return capture(factory.create(arg0, arg1, arg2, arg3, arg4, arg5, arg6));
}
/**
* Creates a new object being managed by this scope.
* @param factory constructor
* @return new scoped object
*/
public O create(Factory8 factory, A arg0, B arg1, C arg2, D arg3, E arg4, F arg5, G arg6, H arg7) {
return capture(factory.create(arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7));
}
/**
* Creates a new object being managed by this scope.
* @param factory constructor
* @return new scoped object
*/
public O create(Factory9 factory, A arg0, B arg1, C arg2, D arg3, E arg4, F arg5, G arg6, H arg7, I arg8) {
return capture(factory.create(arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8));
}
/**
* Creates a new object being managed by this scope with cleanup function.
* @param cleanup the destroyer function
* @param factory constructor
* @return new scoped object
*/
public O create(Consumer cleanup, Factory factory) {
return capture(cleanup, factory.create());
}
/**
* Creates a new object being managed by this scope with cleanup function.
* @param cleanup the destroyer function
* @param factory constructor
* @return new scoped object
*/
public O create(Consumer cleanup, Factory1 factory, A arg0) {
return capture(cleanup, factory.create(arg0));
}
/**
* Creates a new object being managed by this scope with cleanup function.
* @param cleanup the destroyer function
* @param factory constructor
* @return new scoped object
*/
public O create(Consumer cleanup, Factory2 factory, A arg0, B arg1) {
return capture(cleanup, factory.create(arg0, arg1));
}
/**
* Creates a new object being managed by this scope with cleanup function.
* @param cleanup the destroyer function
* @param factory constructor
* @return new scoped object
*/
public O create(Consumer cleanup, Factory3 factory, A arg0, B arg1, C arg2) {
return capture(cleanup, factory.create(arg0, arg1, arg2));
}
/**
* Creates a new object being managed by this scope with cleanup function.
* @param cleanup the destroyer function
* @param factory constructor
* @return new scoped object
*/
public O create(Consumer cleanup, Factory4 factory, A arg0, B arg1, C arg2, D arg3) {
return capture(cleanup, factory.create(arg0, arg1, arg2, arg3));
}
/**
* Creates a new object being managed by this scope with cleanup function.
* @param cleanup the destroyer function
* @param factory constructor
* @return new scoped object
*/
public O create(Consumer cleanup, Factory5 factory, A arg0, B arg1, C arg2, D arg3, E arg4) {
return capture(cleanup, factory.create(arg0, arg1, arg2, arg3, arg4));
}
/**
* Creates a new object being managed by this scope with cleanup function.
* @param cleanup the destroyer function
* @param factory constructor
* @return new scoped object
*/
public O create(Consumer cleanup, Factory6 factory, A arg0, B arg1, C arg2, D arg3, E arg4, F arg5) {
return capture(cleanup, factory.create(arg0, arg1, arg2, arg3, arg4, arg5));
}
/**
* Creates a new object being managed by this scope with cleanup function.
* @param cleanup the destroyer function
* @param factory constructor
* @return new scoped object
*/
public O create(Consumer cleanup, Factory7 factory, A arg0, B arg1, C arg2, D arg3, E arg4, F arg5, G arg6) {
return capture(cleanup, factory.create(arg0, arg1, arg2, arg3, arg4, arg5, arg6));
}
/**
* Creates a new object being managed by this scope with cleanup function.
* @param cleanup the destroyer function
* @param factory constructor
* @return new scoped object
*/
public O create(Consumer cleanup, Factory8 factory, A arg0, B arg1, C arg2, D arg3, E arg4, F arg5, G arg6, H arg7) {
return capture(cleanup, factory.create(arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7));
}
/**
* Creates a new object being managed by this scope with cleanup function.
* @param cleanup the destroyer function
* @param factory constructor
* @return new scoped object
*/
public O create(Consumer cleanup, Factory9 factory, A arg0, B arg1, C arg2, D arg3, E arg4, F arg5, G arg6, H arg7, I arg8) {
return capture(cleanup, factory.create(arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8));
}
}