Files
xtree/src/model/xqviewmodel.cpp

601 lines
17 KiB
C++
Raw Normal View History

2025-08-22 22:57:06 +02:00
/***************************************************************************
source::worx xtree
Copyright © 2024-2025 c.holzheuer
christoph.holzheuer@gmail.com
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
***************************************************************************/
#include <QMessageBox>
#include <QUndoStack>
#include <xqexception.h>
#include <xqviewmodel.h>
#include <xqselectionmodel.h>
#include <xqtreetable.h>
#include <xqcommand.h>
#include <xqitemdelegate.h>
#include <xqitemfactory.h>
#include <znode_factory.h>
// create global dummy item as
// fallback return value (klappt nicht)
//Q_GLOBAL_STATIC(XQItem,s_dummyItem)
//! hilfsfunkion, zeigt den string-content() für alle elemente der liste
void showItemList( const XQItemList& list)
{
for(const auto& entry : list )
2025-08-27 15:39:33 +02:00
qDebug() << " --- itemList: " << entry->text();
2025-08-22 22:57:06 +02:00
qDebug();
}
2025-09-02 16:58:56 +02:00
//! Konstruktor mit parent.
2025-08-22 22:57:06 +02:00
XQViewModel::XQViewModel( QObject* parent )
: QStandardItemModel{ parent }, _itemFactory{ XQItemFactory::instance() }
{
invisibleRootItem()->setData( "[rootItem]", Qt::DisplayRole );
setItemPrototype( new XQItem );
2025-09-03 17:23:52 +02:00
2025-09-02 16:58:56 +02:00
// auf änderungen kann in den unterklassen reagiert werden
connect(this, &QStandardItemModel::itemChanged, this, [this](QStandardItem *item)
2025-09-01 23:28:24 +02:00
{
2025-09-02 16:58:56 +02:00
XQItem* xqItem = static_cast<XQItem*>(item);
2025-09-03 17:23:52 +02:00
emit xqItemChanged( *xqItem );
2025-09-01 23:28:24 +02:00
});
2025-09-03 17:23:52 +02:00
2025-09-01 23:28:24 +02:00
2025-09-02 16:58:56 +02:00
//qRegisterMetaType<XQItem>("XQItem");
2025-08-22 22:57:06 +02:00
}
//! gibt einen static-cast<QXItem*> auf 'invisibleRootItem()' zurück
const XQItem& XQViewModel::xqRootItem()
{
// das ist ein hack, denn 'invisibleRootItem()' ist und bleibt ein
// QStandardItem. Trick: keine eigenen members in XQItem, alles
// dynamisch über den ItemData Mechanismus wie in QStandardItem
return *static_cast<XQItem*>(invisibleRootItem());
2025-09-04 14:56:18 +02:00
}
//! Gibt den daten root node des models zurück.
2025-08-22 22:57:06 +02:00
2025-09-04 14:56:18 +02:00
XQNodePtr XQViewModel::contentRootNode()
{
return _contentRoot;
2025-08-22 22:57:06 +02:00
}
//! hifsfunktion, die das item zu einen index zurückgibt
XQItem& XQViewModel::xqItemFromIndex(const QModelIndex& index) const
{
if( index.isValid() )
{
QStandardItem* xqItem = QStandardItemModel::itemFromIndex(index);
if( xqItem )
return *static_cast<XQItem*>(xqItem);
}
return XQItem::fallBackDummyItem();
}
//! hilfsfunktiom, die das erste xqitem einer zeile zurückgibt.
XQItem& XQViewModel::xqFirstItem(int row) const
{
return *static_cast<XQItem*>( QStandardItemModel::item(row) );
}
2025-09-01 17:40:08 +02:00
void XQViewModel::expandNewItem(const QModelIndex& index)
{
if( _treeTable )
{
// ... ausklappen...
_treeTable->expand( index );
// ... und markieren
_treeTable->setCurrentIndex( index );
}
}
2025-08-22 22:57:06 +02:00
//! initialisiert dieses model über den namen. Es wird hier
//! nur die strukur erzeugt, keine inhalte.
void XQViewModel::initModel(const QString& modelName)
{
/*
model
section
header
data
section
...
*/
2025-08-23 14:37:36 +02:00
setObjectName( modelName );
qDebug() << " --- initModel: " << objectName();
2025-08-22 22:57:06 +02:00
// model rootnode finden -> <DocumentTreeModel>
XQNodePtr modelSheet = _itemFactory.findModelSheet( modelName ); // throws
// #1: über alle sections
for( auto& sectionNode : modelSheet->children() )
{
// #2: (optionalen?) header erzeugen
const XQNodePtr header = sectionNode->find_child_by_tag_name( c_Header );
if( header )
{
2025-08-27 14:06:31 +02:00
XQItemList list = _itemFactory.makeRow( header, nullptr );
2025-08-22 22:57:06 +02:00
addSection(list, sectionNode );
}
}
}
//! Hilfsfunktion: fügt die item-liste unserem model hinzu und erzeugt eine 'section'.
//! die section kann erst gültig sein, wenn die items im model gelandet sind,
//! deswegen ist das hier zusammengefasst.
//! Wrzeugt dann eine section aus einer frisch erzeugten itemlist. Der erste modelindex
//! der liste und der root knoten der model-beschreibung werden gespeichert.
2025-09-06 11:08:07 +02:00
void XQViewModel::addSection(const XQItemList& list, const XQNodePtr& sheetNode )
2025-08-22 22:57:06 +02:00
{
// 1. die liste darf nicht leer sein
Q_ASSERT(!list.isEmpty());
2025-09-06 11:08:07 +02:00
// 2. sheetNode muss da sein
Q_ASSERT(sheetNode);
2025-08-22 22:57:06 +02:00
// 3. 'ContenType' muss vorhanden sein
2025-09-06 11:08:07 +02:00
if( !sheetNode->has_attribute( c_ContentType) )
2025-08-22 22:57:06 +02:00
throw XQException( "section list: Section node needs attribute 'ContentType'!");
// 5. das erzeugt dann auch valide indices
appendRow(list);
2025-09-06 11:08:07 +02:00
const QString &sectionKey = sheetNode->attribute(c_ContentType);
2025-08-22 22:57:06 +02:00
// 6. jetzt können wir auch die sction erzeugen
2025-09-06 11:08:07 +02:00
const XQModelSection& section = _sections.createSection( sectionKey, list[0]->index(), sheetNode );
2025-08-22 22:57:06 +02:00
// ... und es der welt mitteilen.
emit sectionCreated( section );
}
2025-09-05 21:42:40 +02:00
//! SLOT, toggled die section mit dem 'sectionKey' (hier: contentType)
2025-09-01 17:40:08 +02:00
void XQViewModel::onToggleSection(const QString& sectionKey )
{
2025-09-05 21:42:40 +02:00
toggleSection( _sections.sectionByKey(sectionKey) );
}
2025-09-01 17:40:08 +02:00
2025-09-05 21:42:40 +02:00
//! toggled die gegebene model section.
void XQViewModel::toggleSection( const XQModelSection& section )
{
qDebug() << " --- toggleSection: " << section.contentType();
if( section.isValid() && _treeTable )
{
XQSectionRange pos = _sections.sectionRange(section);
2025-09-05 17:12:38 +02:00
//int fstRow = _sections.firstRow(index);
//int lstRow = _sections.lastRow(index);
2025-09-06 11:08:07 +02:00
_treeTable->toggleRowsHidden(pos.firstRow, pos.lastRow );
qDebug() << " --- toggleSection: " << section.contentType();
2025-09-04 17:01:01 +02:00
2025-09-05 21:42:40 +02:00
// hier nicht!?
2025-09-06 11:08:07 +02:00
emit sectionToggled(section);
2025-09-04 17:01:01 +02:00
2025-09-05 21:42:40 +02:00
}
2025-09-04 17:01:01 +02:00
}
2025-09-05 17:12:38 +02:00
2025-09-02 16:58:56 +02:00
/*
//! SLOT als weiterleitung vom SIGNAL itemchanged
void XQViewModel::onItemChanged(XQItem* item )
{
qDebug() << " --- BASE item changed: " << item->text();
}
*/
2025-08-22 22:57:06 +02:00
//! SLOT, der aufgerufen wird, wenn eine edit-action getriggert wurde.
void XQViewModel::onActionTriggered(QAction* action)
{
qDebug() << " --- onActionTriggered: count:" << XQNode::s_Count;
// all selected indices
QModelIndexList selectionList = treeTable()->selectionModel()->selectedRows();
// extract command type
XQCommand::CmdType cmdType = action->data().value<XQCommand::CmdType>();
switch( cmdType )
{
// just handle undo ...
case XQCommand::cmdUndo :
return _undoStack->undo();
// ... or do/redo
case XQCommand::cmdRedo :
return _undoStack->redo();
// for copy & cut, we create a clone of the dataNodes in the clipboard
case XQCommand::cmdCopy :
case XQCommand::cmdCut :
// don't 'copy' empty selections
if( !selectionList.isEmpty() )
_clipBoard.saveNodes( selectionList );
// for copy, we are done, since copy cannot be undone
if( cmdType == XQCommand::cmdCopy )
return;
default:
break;
}
// we create a command
XQCommand* command = new XQCommand( cmdType, this );
// store the row positions of the selected indices
command->saveNodes( selectionList );
command->setOriginIndex( treeTable()->currentIndex() );
// execute command
_undoStack->push( command );
}
//! führt die 'redo' action des gegebenen commnds aus.
2025-08-26 19:41:28 +02:00
void XQViewModel::onCommandRedo( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
static MemCallMap redoCalls
{
{ XQCommand::cmdToggleSection, &XQViewModel::cmdToggleSection },
{ XQCommand::cmdCut, &XQViewModel::cmdCut },
{ XQCommand::cmdPaste, &XQViewModel::cmdPaste },
{ XQCommand::cmdNew, &XQViewModel::cmdNew },
{ XQCommand::cmdDelete, &XQViewModel::cmdDelete }
};
try
{
MemCall memCall = redoCalls[command.commandType()];
if( memCall )
(this->*memCall)( command );
else
qDebug() << " --- onCommandRedo: default: not handled: " << command.toString();
}
catch( XQException& exception )
{
qDebug() << exception.what();
QMessageBox::critical( nullptr, "Failure", QString("Failure: %1").arg(exception.what()) );
}
}
//! führt die 'undo' action des gegebenen commnds aus.
2025-08-26 19:41:28 +02:00
void XQViewModel::onCommandUndo( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
qDebug() << " --- onCommandUndo: count: " << XQNode::s_Count;
static MemCallMap undoCalls
{
{ XQCommand::cmdToggleSection, &XQViewModel::cmdToggleSection },
{ XQCommand::cmdCut, &XQViewModel::cmdCutUndo },
{ XQCommand::cmdPaste, &XQViewModel::cmdPasteUndo },
{ XQCommand::cmdNew, &XQViewModel::cmdNewUndo },
{ XQCommand::cmdDelete, &XQViewModel::cmdDeleteUndo },
};
try
{
MemCall memCall = undoCalls[command.commandType()];
if( memCall )
(this->*memCall)( command );
else
qDebug() << " --- onCommandUndo: default: not handled: " << command.toString();
}
catch( XQException& exception )
{
qDebug() << exception.what();
QMessageBox::critical( nullptr, "Failure", QString("Failure: %1").arg(exception.what()) );
}
}
// undo-/redo-able stuff
//! markierte knoten entfernen, 'command' enthält die liste
2025-08-26 19:41:28 +02:00
void XQViewModel::cmdCut( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
// wir gehen rückwärts über alle gemerkten knoten ...
for (auto it = command.rbegin(); it != command.rend(); ++it)
{
// ... holen das erste item, das auch den content node enthält
//const XQNodeBackup& entry = *it;
// jetzt löschen, dabei wird die parent-verbindung entfernt
const XQNodeBackup& entry = *it;
XQItem& firstItem = xqFirstItem( (*it).itemPos );
qDebug() << " --- Cut: " << firstItem.text() << " " << firstItem.row() << " id#" << entry.contentNode->_id;
entry.contentNode->unlink_self();
removeRow(entry.itemPos );
}
}
//! entfernte knoten wieder einfügen , 'command' enthält die liste
2025-08-26 19:41:28 +02:00
void XQViewModel::cmdCutUndo( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
// die anfangsposition
int itmPos = command.first().itemPos;
// die 'zuständige' section rausfinden
2025-09-05 11:49:36 +02:00
const XQModelSection& section = _sections.sectionByRow( itmPos );
2025-08-22 22:57:06 +02:00
// über alle einträge ...
for (auto& entry : command )
{
const XQNodePtr& savedNode = entry.contentNode;
// __fix! should not be _contentRoot!
savedNode->add_me_at( entry.nodePos, _contentRoot );
2025-08-27 14:06:31 +02:00
XQItemList list = _itemFactory.makeRow( section.sheetRootNode(), savedNode );
2025-08-22 22:57:06 +02:00
XQItem& firstItem = *((XQItem*)list[0]);
qDebug() << " --- Cut Undo: " << firstItem.text() << " " << firstItem.row() << " id#" << entry.contentNode->_id << " count: " << entry.contentNode.use_count();
insertRow( entry.itemPos, list );
}
}
//! clipboard inhalte einfügen
2025-08-26 19:41:28 +02:00
void XQViewModel::cmdPaste( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
// selection holen ...
QItemSelectionModel* selectionModel = treeTable()->selectionModel();
// ... und löschen
selectionModel->clearSelection();
// aktuelles item finden
const XQItem& item = xqItemFromIndex( command.originIndex() );
// die neue item position ist unter dem akutellen item
int insRow = item.row()+1;
int nodePos = item.contentNode()->own_pos()+1;
// die zugehörige section finden
2025-09-05 11:49:36 +02:00
const XQModelSection& section = _sections.sectionByRow( insRow-1 );
2025-08-22 22:57:06 +02:00
// wir pasten das clipboard
for (auto& entry : _clipBoard )
{
2025-08-24 14:19:39 +02:00
// noch einen clone vom clone im clipboard erzeugen ...
2025-08-22 22:57:06 +02:00
XQNodePtr newNode = entry.contentNode->clone(section.contentRootNode() );
2025-08-24 14:19:39 +02:00
// ... diesen einfügen ...
newNode->add_me_at( nodePos );
2025-08-22 22:57:06 +02:00
// ... und damit eine frische item-row erzeugen
2025-08-27 14:06:31 +02:00
XQItemList list = _itemFactory.makeRow( section.sheetRootNode(), newNode );
2025-08-22 22:57:06 +02:00
insertRow( insRow, list );
// die neue item-row selektieren
const QModelIndex& selIdx = list[0]->index();
_treeTable->selectionModel()->select(selIdx, QItemSelectionModel::Select | QItemSelectionModel::Rows);
// zur nächsten zeile
insRow++;
nodePos++;
}
// unsere änderungen merken fürs 'undo'
2025-08-26 19:41:28 +02:00
/// fix_xx
const_cast<XQCommand&>(command).saveNodes( selectionModel->selectedRows() );
2025-08-22 22:57:06 +02:00
}
//! einfügen aus dem clipboard wieder rückgängig machen
2025-08-26 19:41:28 +02:00
void XQViewModel::cmdPasteUndo( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
command.dumpList("Paste UNDO");
// wir gehen rückwärts über alle markieren knoten ...
for (auto it = command.rbegin(); it != command.rend(); ++it)
{
// ... holen das erste item, das auch den content node enthält
const XQNodeBackup& entry = *it;
XQItem& firstItem = xqFirstItem( (*it).itemPos );
qDebug() << " --- Cut: " << firstItem.text() << " " << firstItem.row();
// jetzt löschen
entry.contentNode->unlink_self();
removeRow(entry.itemPos );
}
}
// don't clone into clipboard, remove items
//! entfernen der selection ohne copy in clipboard.
2025-08-26 19:41:28 +02:00
void XQViewModel::cmdDelete( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
// wir gehen rückwärts über alle markieren knoten ...
for (auto it = command.rbegin(); it != command.rend(); ++it)
{
// ... holen das erste item, das auch den content node enthält
const XQNodeBackup& entry = *it;
XQItem& firstItem = xqFirstItem( (*it).itemPos );
2025-09-06 11:08:07 +02:00
qDebug() << " --- delete: " << firstItem.text() << " " << firstItem.row();
2025-08-22 22:57:06 +02:00
// jetzt löschen
entry.contentNode->unlink_self();
removeRow(entry.itemPos );
}
}
2025-09-04 13:52:23 +02:00
//! macht 'delete' wieder rückgängig.
2025-08-22 22:57:06 +02:00
2025-08-26 19:41:28 +02:00
void XQViewModel::cmdDeleteUndo( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
2025-09-06 11:08:07 +02:00
for (const auto& entry : command)
{
qDebug() << " --- delete UNDo: " << entry.contentNode->to_string();
}
2025-08-22 22:57:06 +02:00
}
//! legt eine neue, leere zeile an.
2025-08-26 19:41:28 +02:00
void XQViewModel::cmdNew( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
const QModelIndex& origin = command.originIndex();
2025-09-04 13:52:23 +02:00
XQItem& target = xqItemFromIndex( origin );
2025-08-22 22:57:06 +02:00
// current data node
2025-09-04 13:52:23 +02:00
XQNodePtr node = target.contentNode();
2025-09-04 18:10:14 +02:00
// we create a new data node
2025-09-04 17:01:01 +02:00
XQNodePtr newNode = XQNode::make_node( node->tag_name(), node->tag_value() );
2025-08-22 22:57:06 +02:00
// store node in node->parent()
2025-09-04 14:56:18 +02:00
newNode->add_me_at( node->own_pos(), node->parent() );
2025-09-04 17:01:01 +02:00
//...
2025-09-05 21:42:40 +02:00
const XQModelSection& section = _sections.sectionByRow( origin.row() );
2025-08-22 22:57:06 +02:00
2025-09-04 17:01:01 +02:00
// neue, leere zeile erzeugen ...
2025-09-04 13:52:23 +02:00
XQItemList list =_itemFactory.makeRow( section.sheetRootNode(), newNode );
2025-09-04 17:01:01 +02:00
// ... zur treeview hinzufügen ...
2025-08-22 22:57:06 +02:00
insertRow( origin.row(), list );
2025-09-04 17:01:01 +02:00
// ... editierbar machen ...
QModelIndex newIndex = list[0]->index();
treeTable()->setCurrentIndex( newIndex );
treeTable()->edit( newIndex );
// ,,, und fürs undo speichern
const_cast<XQCommand&>(command).saveNodes( {newIndex} );
2025-09-01 17:40:08 +02:00
}
2025-09-04 13:52:23 +02:00
2025-08-22 22:57:06 +02:00
//! entfernt die neu angelegte zeile.
2025-08-26 19:41:28 +02:00
void XQViewModel::cmdNewUndo( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
2025-09-04 17:01:01 +02:00
cmdDelete( command );
2025-08-22 22:57:06 +02:00
}
//! schaltet eine section sichtbar oder unsichtbar.
2025-08-26 19:41:28 +02:00
void XQViewModel::cmdToggleSection( const XQCommand& command )
2025-08-22 22:57:06 +02:00
{
const QModelIndex& index = command.originIndex();
Q_ASSERT(index.isValid());
2025-09-05 21:42:40 +02:00
toggleSection( _sections.sectionByRow(index.row()) );
2025-08-22 22:57:06 +02:00
}
2025-09-05 21:42:40 +02:00
2025-08-22 22:57:06 +02:00
2025-09-01 17:40:08 +02:00
//! gibt die treetable zurück
2025-08-22 22:57:06 +02:00
XQTreeTable* XQViewModel::treeTable()
{
return _treeTable;
}
//! setzt die treetable als member.
void XQViewModel::setTreeTable(XQTreeTable* mainView )
{
// store view for direct access: the maintree
_treeTable = mainView;
2025-09-02 16:58:56 +02:00
// set myself as model to the mainview
2025-08-22 22:57:06 +02:00
_treeTable->setModel(this);
XQItemDelegate* delegate = new XQItemDelegate( *this );
_treeTable->setItemDelegate( delegate );
_contextMenu = new XQContextMenu( mainView );
connect( _treeTable, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(onShowContextMenu(QPoint)));
//connect( _treeTable, SIGNAL(doubleClicked(QModelIndex)), this, SLOT(onDoubleClicked(QModelIndex)) );
connect(_contextMenu, SIGNAL(triggered(QAction*)), this, SLOT(onActionTriggered(QAction*)));
// __fixme, die view soll über das modelsheet konfiguriert werden!
setupViewProperties();
}
//! setzt die eigenschaften der TreeTable.
void XQViewModel::setupViewProperties()
{
_treeTable->setContextMenuPolicy(Qt::CustomContextMenu);
_treeTable->setEditTriggers(QAbstractItemView::DoubleClicked | QAbstractItemView::EditKeyPressed);
_treeTable->setSelectionBehavior(QAbstractItemView::SelectRows);
_treeTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
//_treeTable->setSelectionMode(QAbstractItemView::ContiguousSelection);
_treeTable->setSelectionModel( new XQSelectionModel(this) );
}
//! gibt den undo-stack zurück.
QUndoStack* XQViewModel::undoStack()
{
return _undoStack;
}
//! setzt den undo-stack.
void XQViewModel::setUndoStack( QUndoStack* undoStack )
{
_undoStack = undoStack;
}
//! SLOT, der die erstellung & anzeige es context-menues triggert.
void XQViewModel::onShowContextMenu(const QPoint& point)
{
initContextMenu();
_contextMenu->popup(_treeTable->mapToGlobal(point));
}
//! gibt die namen der neuen data-roles zurück.
//! __fix: die alten roles fehlen hier!
QHash<int, QByteArray> XQViewModel::roleNames() const
{
2025-08-24 09:44:51 +02:00
QHash<int, QByteArray> roles = QStandardItemModel::roleNames();
2025-08-22 22:57:06 +02:00
roles[XQItem::ContentRole] = "content";
roles[XQItem::ItemTypeRole] = "itemType";
roles[XQItem::RenderStyleRole] = "renderStyle";
roles[XQItem::EditorTypeRole] = "editorType";
roles[XQItem::UnitTypeRole] = "unitType";
roles[XQItem::FixedChoicesRole] = "fixedChoices";
roles[XQItem::ContentNodeRole] = "contentNode";
roles[XQItem::SheetNodeRole] = "sheetNode";
roles[XQItem::TypeKeyRole] = "typeKey";
return roles;
}