MessageList::Core::ModelInvariantRowMapper

Search for usage in LXR

MessageList::Core::ModelInvariantRowMapper Class Reference

#include <modelinvariantrowmapper.h>

Inheritance diagram for MessageList::Core::ModelInvariantRowMapper:

Public Member Functions

void createModelInvariantIndex (int modelIndexRow, ModelInvariantIndex *invariantToFill)
 
QList< ModelInvariantIndex * > * modelIndexRowRangeToModelInvariantIndexList (int startIndexRow, int count)
 
ModelInvariantIndexmodelIndexRowToModelInvariantIndex (int modelIndexRow)
 
int modelInvariantIndexToModelIndexRow (ModelInvariantIndex *invariant)
 
void modelReset ()
 
void modelRowsInserted (int modelIndexRowPosition, int count)
 
QList< ModelInvariantIndex * > * modelRowsRemoved (int modelIndexRowPosition, int count)
 
void setLazyUpdateChunkInterval (int chunkInterval)
 
void setLazyUpdateIdleInterval (int idleInterval)
 
- Public Member Functions inherited from QObject
 QObject (QObject *parent)
 
QBindable< QStringbindableObjectName ()
 
bool blockSignals (bool block)
 
const QObjectListchildren () const const
 
QMetaObject::Connection connect (const QObject *sender, const char *signal, const char *method, Qt::ConnectionType type) const const
 
void deleteLater ()
 
void destroyed (QObject *obj)
 
bool disconnect (const char *signal, const QObject *receiver, const char *method) const const
 
bool disconnect (const QObject *receiver, const char *method) const const
 
void dumpObjectInfo () const const
 
void dumpObjectTree () const const
 
QList< QByteArraydynamicPropertyNames () const const
 
virtual bool event (QEvent *e)
 
virtual bool eventFilter (QObject *watched, QEvent *event)
 
findChild (const QString &name, Qt::FindChildOptions options) const const
 
QList< T > findChildren (const QRegularExpression &re, Qt::FindChildOptions options) const const
 
QList< T > findChildren (const QString &name, Qt::FindChildOptions options) const const
 
QList< T > findChildren (Qt::FindChildOptions options) const const
 
bool inherits (const char *className) const const
 
void installEventFilter (QObject *filterObj)
 
bool isQuickItemType () const const
 
bool isWidgetType () const const
 
bool isWindowType () const const
 
void killTimer (int id)
 
virtual const QMetaObjectmetaObject () const const
 
void moveToThread (QThread *targetThread)
 
QString objectName () const const
 
void objectNameChanged (const QString &objectName)
 
QObjectparent () const const
 
QVariant property (const char *name) const const
 
 Q_CLASSINFO (Name, Value)
 
 Q_EMIT Q_EMIT
 
 Q_ENUM (...)
 
 Q_ENUM_NS (...)
 
 Q_ENUMS (...)
 
 Q_FLAG (...)
 
 Q_FLAG_NS (...)
 
 Q_FLAGS (...)
 
 Q_GADGET Q_GADGET
 
 Q_GADGET_EXPORT (EXPORT_MACRO)
 
 Q_INTERFACES (...)
 
 Q_INVOKABLE Q_INVOKABLE
 
 Q_MOC_INCLUDE Q_MOC_INCLUDE
 
 Q_NAMESPACE Q_NAMESPACE
 
 Q_NAMESPACE_EXPORT (EXPORT_MACRO)
 
 Q_OBJECT Q_OBJECT
 
 Q_PROPERTY (...)
 
 Q_REVISION Q_REVISION
 
 Q_SET_OBJECT_NAME (Object)
 
 Q_SIGNAL Q_SIGNAL
 
 Q_SIGNALS Q_SIGNALS
 
 Q_SLOT Q_SLOT
 
 Q_SLOTS Q_SLOTS
 
qobject_cast (const QObject *object)
 
qobject_cast (QObject *object)
 
 QT_NO_NARROWING_CONVERSIONS_IN_CONNECT QT_NO_NARROWING_CONVERSIONS_IN_CONNECT
 
void removeEventFilter (QObject *obj)
 
void setObjectName (const QString &name)
 
void setObjectName (QAnyStringView name)
 
void setParent (QObject *parent)
 
bool setProperty (const char *name, const QVariant &value)
 
bool setProperty (const char *name, QVariant &&value)
 
bool signalsBlocked () const const
 
int startTimer (int interval, Qt::TimerType timerType)
 
int startTimer (std::chrono::milliseconds interval, Qt::TimerType timerType)
 
QThreadthread () const const
 

Additional Inherited Members

- Public Types inherited from QObject
typedef  QObjectList
 
- Properties inherited from QObject
 objectName
 
- Static Public Member Functions inherited from QObject
QMetaObject::Connection connect (const QObject *sender, const char *signal, const QObject *receiver, const char *method, Qt::ConnectionType type)
 
QMetaObject::Connection connect (const QObject *sender, const QMetaMethod &signal, const QObject *receiver, const QMetaMethod &method, Qt::ConnectionType type)
 
QMetaObject::Connection connect (const QObject *sender, PointerToMemberFunction signal, const QObject *context, Functor functor, Qt::ConnectionType type)
 
QMetaObject::Connection connect (const QObject *sender, PointerToMemberFunction signal, const QObject *receiver, PointerToMemberFunction method, Qt::ConnectionType type)
 
QMetaObject::Connection connect (const QObject *sender, PointerToMemberFunction signal, Functor functor)
 
bool disconnect (const QMetaObject::Connection &connection)
 
bool disconnect (const QObject *sender, const char *signal, const QObject *receiver, const char *method)
 
bool disconnect (const QObject *sender, const QMetaMethod &signal, const QObject *receiver, const QMetaMethod &method)
 
bool disconnect (const QObject *sender, PointerToMemberFunction signal, const QObject *receiver, PointerToMemberFunction method)
 
QString tr (const char *sourceText, const char *disambiguation, int n)
 
- Protected Member Functions inherited from QObject
virtual void childEvent (QChildEvent *event)
 
virtual void connectNotify (const QMetaMethod &signal)
 
virtual void customEvent (QEvent *event)
 
virtual void disconnectNotify (const QMetaMethod &signal)
 
bool isSignalConnected (const QMetaMethod &signal) const const
 
int receivers (const char *signal) const const
 
QObjectsender () const const
 
int senderSignalIndex () const const
 
virtual void timerEvent (QTimerEvent *event)
 

Detailed Description

This class is an optimizing helper for dealing with large flat QAbstractItemModel objects.

The problem:

When you're an user of a flat QAbstractItemModel you access its contents by the means of QModelIndex. The model index is basically a row index (for flat models). You usually fetch some data for a row and then store the row index in order to fetch more data later (think of a tree view that shows the "name" for an item but when clicked needs to open a window with the details associated to the item).

The problem is that your row indexes may become invalid when rows are added or removed from the model. For instance, if a row is added at position 10, then any cached index after position 10 must be updated in order to point to the same content. With very large models this can become a problem since the update must be somewhat "atomic" in order to preserve consistency. Atomic, then, means "unbreakable": you can't chunk it in smaller pieces. This also means that your application will simply freeze if you have a model with 100000 cached indexes and a row is added/removed at the beginning. Yet worse: your app will freeze EVERY time a row is added. This means that if you don't really optimize, or just add non contiguous rows, you must do the update multiple times...

This class tries to solve this problem with a little overhead. It basically gives you a ModelInvariantIndex for each "new" row you query. Later the model may change by addition/removal of rows but with a ModelInvariantIndex you will still be able to retrieve the content that the index was pointing to at the time it was created.

You don't need to implement any row update in your rowsInserted() / rowsRemoved() handlers. Just call the modelRowsInserted() modelRowsRemoved() functions: they will do everything for you in a substantially constant time.

As the model structure changes the lookups will get a bit slower since the row mapper must apply the changes sequentially to each invariant. To avoid this, and to avoid storing the whole history of changes the ModelInvariantRowMapper will perform a background update of your ModelInvariantIndex objects. You don't need to care about this: it will happen automagically.

The ModelInvariantIndex allocation and destruction is in fact left to you. This is a GOOD approach because you're very likely to have some sort of structures associated to the items you display. You may then simply derive your objects from ModelInvariantIndex. This will save an additional memory allocation for each one of your items (we are optimizing, aren't we ?) and you will be able to dynamic_cast<> the ModelInvariantIndex pointers directly to your structure pointers (which will likely save a large QHash<>...). Even in the unlikely case in that you don't have a data structure for your items, it's still an operator new() call that YOU make instead of this implementation. It doesn't impact performance at all. You just have to remember to delete your ModelInvariantIndex objects when you no longer need them.

Definition at line 75 of file modelinvariantrowmapper.h.

Constructor & Destructor Documentation

◆ ModelInvariantRowMapper()

ModelInvariantRowMapper::ModelInvariantRowMapper ( )
explicit

Definition at line 50 of file modelinvariantrowmapper.cpp.

◆ ~ModelInvariantRowMapper()

ModelInvariantRowMapper::~ModelInvariantRowMapper ( )
override

Definition at line 66 of file modelinvariantrowmapper.cpp.

Member Function Documentation

◆ createModelInvariantIndex()

void ModelInvariantRowMapper::createModelInvariantIndex ( int modelIndexRow,
ModelInvariantIndex * invariantToFill )

Binds a ModelInvariantIndex structure to the specified CURRENT modelIndexRow.

Later you can use the ModelInvariantIndex to retrieve the model contents that the parameter modelIndexRow refers to... even if the model changes. Call this function only if you're sure that there is no such invariant yet, otherwise take a look at modelIndexRowToModelInvariantIndex().

This function ASSUMES that invariantToFill is a newly allocated ModelInvariantIndex.

Definition at line 326 of file modelinvariantrowmapper.cpp.

◆ modelIndexRowRangeToModelInvariantIndexList()

QList< ModelInvariantIndex * > * ModelInvariantRowMapper::modelIndexRowRangeToModelInvariantIndexList ( int startIndexRow,
int count )

This basically applies modelIndexRowToModelInvariantIndex() to a range of elements.

The returned pointer can be null if no existing ModelInvariantIndex object were present in the range (this can happen if you don't request some of them). If the returned value is not 0 then you're responsible of deleting it.

Definition at line 345 of file modelinvariantrowmapper.cpp.

◆ modelIndexRowToModelInvariantIndex()

ModelInvariantIndex * ModelInvariantRowMapper::modelIndexRowToModelInvariantIndex ( int modelIndexRow)

Finds the existing ModelInvariantIndex that belongs to the specified CURRENT modelIndexRow.

Returns the ModelInvariantIndex found or 0 if such an invariant wasn't yet created (by the means of createModelInvariantIndex()).

Definition at line 340 of file modelinvariantrowmapper.cpp.

◆ modelInvariantIndexToModelIndexRow()

int ModelInvariantRowMapper::modelInvariantIndexToModelIndexRow ( ModelInvariantIndex * invariant)

Maps a ModelInvariantIndex to the CURRENT associated row index in the model.

As long as the underlying model is consistent, the returned row index is guaranteed to point to the content that this ModelInvariantIndex pointed at the time it was created.

Returns the current associated row index in the model or -1 if the invariant does not belong to this mapper (model) or is marked as invalid at all.

Definition at line 234 of file modelinvariantrowmapper.cpp.

◆ modelReset()

void ModelInvariantRowMapper::modelReset ( )

Call this function from your handlers of reset() and layoutChanged() AFTER you ve last accessed the model underlying data.

You probably want this function to be the first call of your reset() or layoutChanged() handlers.

This function assumes that all the ModelInvariantIndex are being invalidated and need to be required.

Definition at line 547 of file modelinvariantrowmapper.cpp.

◆ modelRowsInserted()

void ModelInvariantRowMapper::modelRowsInserted ( int modelIndexRowPosition,
int count )

Call this function when rows are inserted to the underlying model BEFORE scanning the model for the new items.

You probably want this function to be the first call in your rowsInserted() handler or the last call in the rowsAboutToBeInserted() handler.

Definition at line 374 of file modelinvariantrowmapper.cpp.

◆ modelRowsRemoved()

QList< ModelInvariantIndex * > * ModelInvariantRowMapper::modelRowsRemoved ( int modelIndexRowPosition,
int count )

Call this function when rows are removed from the underlying model AFTER accessing the removed rows for the last time.

You probably want this function to be the first call of your rowsRemoved() handler or the last call in the rowsAboutToBeRemoved() handler.

This function will invalidate any ModelInvariantIndex instances that are affected by the change. It will also do you a favor by returning the list of the invalidated ModelInvariantIndex objects since you'll probably want to delete them. The returned pointer can be null if no existing ModelInvariantIndex object were affected by the change (this can happen if you don't request some of them). If the returned value is not 0 then you're responsible of deleting it.

Definition at line 437 of file modelinvariantrowmapper.cpp.

◆ setLazyUpdateChunkInterval()

void ModelInvariantRowMapper::setLazyUpdateChunkInterval ( int chunkInterval)

Sets the maximum time we can spend inside a single lazy update step.

The larger this time, the more resources we consume and leave less to UI processing but also larger the update throughput (that is, we update more items per second). This is 50 msec by default.

Definition at line 224 of file modelinvariantrowmapper.cpp.

◆ setLazyUpdateIdleInterval()

void ModelInvariantRowMapper::setLazyUpdateIdleInterval ( int idleInterval)

Sets the idle time between two lazy updates in milliseconds.

The larger this time, the less resources we consume and leave more to UI processing but also smaller the update throughput (that is, we update less items per second). This is 50 msec by default.

Definition at line 229 of file modelinvariantrowmapper.cpp.


The documentation for this class was generated from the following files:
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Jan 3 2025 11:55:29 by doxygen 1.12.0 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.