Compare commits
	
		
			12 Commits
		
	
	
		
			experiment
			...
			main
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| c4d5155ac3 | |||
| 558ef154a6 | |||
| f70e67e6ed | |||
| b3b37d1ca4 | |||
| f31da16948 | |||
| 1e82128ea0 | |||
| b5f218084b | |||
| f74c004bf9 | |||
| 007970f4ee | |||
| 4996c03b39 | |||
| 6b675cb85e | |||
| defa6a1906 | 
| @@ -33,38 +33,34 @@ TreeView | |||||||
|         //color: TreeView.isSelected ? "#d0eaff" : (row % 2 === 0 ? "#f9f9f9" : "#ffffff") |         //color: TreeView.isSelected ? "#d0eaff" : (row % 2 === 0 ? "#f9f9f9" : "#ffffff") | ||||||
|         //color: TreeView.isSelected ? "#d0eaff" : (row % 2 === 0 ? "#f9f9f9" : "#ffffff") |         //color: TreeView.isSelected ? "#d0eaff" : (row % 2 === 0 ? "#f9f9f9" : "#ffffff") | ||||||
|  |  | ||||||
|  |         //color: index % 2 === 0 ? "#f9f9f9" : "#e0e0e0" | ||||||
|  |         //color: TreeView.isSelected ? "#d0eaff" : (row % 2 === 0 ? "#f9f9f9" : "#ffffff") | ||||||
|  |  | ||||||
|         TextField |         TextField | ||||||
|         { |         { | ||||||
|             id: currentEntry |             id: entry | ||||||
|             anchors.centerIn: parent |             anchors.centerIn: parent | ||||||
|             text:  display |             text:  display | ||||||
|             font.pixelSize: 12 |             font.pixelSize: 11 | ||||||
|  |             horizontalAlignment: Text.AlignHCenter | ||||||
|  |             verticalAlignment: Text.AlignVCenter | ||||||
|  |  | ||||||
|  |             /* | ||||||
|  |             background: Rectangle | ||||||
|  |             { | ||||||
|  |                 //color: entry.enabled ? "transparent" : "#353637" | ||||||
|  |                 //border.color: entry.enabled ? "#21be2b" : "transparent" | ||||||
|  |                 border.color: "transparent" | ||||||
|  |                 color: TreeView.isSelected ? "#d0eaff" : (row % 2 === 0 ? "#f9f9f9" : "#ffffff") | ||||||
|  |  | ||||||
|             // Ändere die Border-Farbe je nachdem, ob das Feld den Fokus hat |  | ||||||
|             property color borderColor: currentEntry.activeFocus ? "dodgerblue" : "gray" |  | ||||||
|             property int borderWidth: currentEntry.activeFocus ? 2 : 1 |  | ||||||
|  |  | ||||||
|             background: Rectangle { |  | ||||||
|                 // Die Farbe des Hintergrunds im Normalzustand |  | ||||||
|                 color: "#ffffff" |  | ||||||
|  |  | ||||||
|                 // Hier werden Rahmenfarbe und -breite definiert |  | ||||||
|                 border.color: currentEntry.borderColor |  | ||||||
|                 border.width: currentEntry.borderWidth |  | ||||||
|  |  | ||||||
|                 // Abgerundete Ecken für ein modernes Aussehen |  | ||||||
|                 radius: 4 |  | ||||||
|  |  | ||||||
|                 // Sanfter Übergang der Rahmenfarbe und -breite |  | ||||||
|                 Behavior on border.color { |  | ||||||
|                     ColorAnimation { duration: 150 } |  | ||||||
|                 } |  | ||||||
|                 Behavior on border.width { |  | ||||||
|                     NumberAnimation { duration: 150 } |  | ||||||
|             } |             } | ||||||
|  |             */ | ||||||
|  |  | ||||||
|  |             // Hintergrund des Textfeldes anpassen | ||||||
|  |             background: Rectangle | ||||||
|  |             { | ||||||
|  |                 //color: TreeView.isSelected ? "#d0eaff" : (row % 2 === 0 ? "#f9f9f9" : "#ffffff") | ||||||
|  |                 color: entry.activeFocus ? "#d0eaff" : (row % 2 === 0 ? "#f9f9f9" : "#ffffff") | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             onEditingFinished: |             onEditingFinished: | ||||||
|   | |||||||
| @@ -31,6 +31,22 @@ XQChildModel::XQChildModel( QObject *parent ) | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | //! Erzeugt eine model-section und fügt den zugehörigen header ein. | ||||||
|  |  | ||||||
|  | void XQChildModel::addSectionEntry( const QString& key, const XQNodePtr& contentEntry ) | ||||||
|  | { | ||||||
|  |   const XQModelSection& section = _sections.sectionByKey( key ); | ||||||
|  |   if(section.isValid() ) | ||||||
|  |   { | ||||||
|  |     section.setContentRootNode( contentEntry->parent() ); | ||||||
|  |     int newRow =_sections.lastRow(section); | ||||||
|  |     XQNodePtr sheetNode = section.sheetRootNode(); | ||||||
|  |     XQItemList list = _itemFactory.makeRow( sheetNode, nullptr ); | ||||||
|  |     insertRow( newRow, list); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| //! erzegt den sichtbaren inhalt des models aus einem root-datenknoten. | //! erzegt den sichtbaren inhalt des models aus einem root-datenknoten. | ||||||
|  |  | ||||||
| void XQChildModel::addModelData( const XQNodePtr& contentRoot ) | void XQChildModel::addModelData( const XQNodePtr& contentRoot ) | ||||||
| @@ -46,7 +62,6 @@ void XQChildModel::addModelData( const XQNodePtr& contentRoot ) | |||||||
|     // Das ist hier der Typ des Eintrags: Panel, Battery ... |     // Das ist hier der Typ des Eintrags: Panel, Battery ... | ||||||
|     QString key = contentEntry->tag_name(); |     QString key = contentEntry->tag_name(); | ||||||
|  |  | ||||||
|  |  | ||||||
|     // 'silent failure' hier der Datenbaum kann auch Knoten enthalten |     // 'silent failure' hier der Datenbaum kann auch Knoten enthalten | ||||||
|     // die nicht für uns gedacht sind. |     // die nicht für uns gedacht sind. | ||||||
|     if (!_sections.hasValidSection(key)) |     if (!_sections.hasValidSection(key)) | ||||||
| @@ -65,12 +80,18 @@ void XQChildModel::addModelData( const XQNodePtr& contentRoot ) | |||||||
|     // _hinter_ der letzen zeile einfügen |     // _hinter_ der letzen zeile einfügen | ||||||
|     insertRow( newRow+1, list); |     insertRow( newRow+1, list); | ||||||
|  |  | ||||||
|  |     // Es können noch Kind-Knoten mit | ||||||
|  |     // Detail-Infos vorhanden sein. | ||||||
|  |  | ||||||
|     if( contentEntry->has_children()) |     if( contentEntry->has_children()) | ||||||
|     { |     { | ||||||
|       qDebug() << " --- AddModelData: CHILD Found for: :" << contentEntry->tag_name() << " sheet parent: " << sheetNode->tag_name(); |       // Die Beschreibungen für die optionalen Kinder | ||||||
|       if( !sheetNode->has_children() ) |       // steck im ersten Knoten des Model-Sheets | ||||||
|         qDebug() << " --- AUA"; |       if( !sheetNode->first_child()->has_children() ) | ||||||
|       //else |         continue; | ||||||
|  |  | ||||||
|  |       _itemFactory.makeChildRow( list.front(), sheetNode->first_child(), contentEntry ); | ||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   } // for |   } // for | ||||||
| @@ -78,22 +99,6 @@ void XQChildModel::addModelData( const XQNodePtr& contentRoot ) | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| //! Erzeugt eine model-section und fügt den zugehörigen header ein. |  | ||||||
|  |  | ||||||
| void XQChildModel::addSectionEntry( const QString& key, const XQNodePtr& contentEntry ) |  | ||||||
| { |  | ||||||
|   const XQModelSection& section = _sections.sectionByKey( key ); |  | ||||||
|   if(section.isValid() ) |  | ||||||
|   { |  | ||||||
|     section.setContentRootNode( contentEntry->parent() ); |  | ||||||
|     int newRow =_sections.lastRow(section); |  | ||||||
|     XQNodePtr sheetNode = section.sheetRootNode(); |  | ||||||
|     XQItemList list = _itemFactory.makeRow( sheetNode, nullptr ); |  | ||||||
|     insertRow( newRow, list); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| //! erzeugt ein adhoc-contextmenu, je nachdem welche aktionen gerade möglich sind. | //! erzeugt ein adhoc-contextmenu, je nachdem welche aktionen gerade möglich sind. | ||||||
|  |  | ||||||
| void XQChildModel::initContextMenu() | void XQChildModel::initContextMenu() | ||||||
|   | |||||||
| @@ -71,7 +71,7 @@ XQItem* XQMainModel::addProjectItem( XQNodePtr contentNode ) | |||||||
|  |  | ||||||
| void XQMainModel::addSectionItem( const XQModelSection& section, XQItem* projectItem ) | void XQMainModel::addSectionItem( const XQModelSection& section, XQItem* projectItem ) | ||||||
| { | { | ||||||
|   XQNodePtr sheetNode = projectItem->sheetNode()->find_child_by_tag_name("CurrentSection"); |   XQNodePtr sheetNode = projectItem->sheetNode()->child("CurrentSection"); | ||||||
|   XQItem* newItem = _itemFactory.makeSingleItem( sheetNode,  section.contentType() ); |   XQItem* newItem = _itemFactory.makeSingleItem( sheetNode,  section.contentType() ); | ||||||
|   projectItem->appendRow( newItem ); |   projectItem->appendRow( newItem ); | ||||||
|   expandNewItem(projectItem->index() ); |   expandNewItem(projectItem->index() ); | ||||||
|   | |||||||
| @@ -62,7 +62,6 @@ void XQMainWindow::setupWorkingDir() | |||||||
| void XQMainWindow::initMainWindow() | void XQMainWindow::initMainWindow() | ||||||
| { | { | ||||||
|  |  | ||||||
|   qDebug() << " --- initMainWindow(): here we go!"; |  | ||||||
|   // das working dir setzen: 'xml' muss als unterverzeichnis vorhanden sein. |   // das working dir setzen: 'xml' muss als unterverzeichnis vorhanden sein. | ||||||
|   setupWorkingDir(); |   setupWorkingDir(); | ||||||
|  |  | ||||||
| @@ -119,10 +118,10 @@ void XQMainWindow::initMainWindow() | |||||||
|     loadDocument( c_DocumentFileName1 ); |     loadDocument( c_DocumentFileName1 ); | ||||||
|     loadDocument( c_DocumentFileName2, true ); |     loadDocument( c_DocumentFileName2, true ); | ||||||
|     //loadDocument( c_DocumentFileName2 ); |     //loadDocument( c_DocumentFileName2 ); | ||||||
|     //loadDocument( c_DocumentFileName3 ); |     loadDocument( c_DocumentFileName3 ); | ||||||
|  |  | ||||||
|  |  | ||||||
|     qDebug() << " --- all here: " << XQNode::s_Count; |     qDebug() << " --- #nodes created: " << XQNode::s_Count; | ||||||
|  |  | ||||||
|   }   |   }   | ||||||
|   catch( XQException& exception ) |   catch( XQException& exception ) | ||||||
| @@ -260,7 +259,6 @@ void XQMainWindow::onTreeViewItemClicked( const XQItem& item ) | |||||||
|  |  | ||||||
| void XQMainWindow::onTreeViewItemChanged(const XQItem& item ) | void XQMainWindow::onTreeViewItemChanged(const XQItem& item ) | ||||||
| { | { | ||||||
|   qDebug() << " --- TREE VIEW itemChanged: text" << item.text() << " parent: " << item.parent()->text()  << " type: " << item.itemType().text() << " : " << (void*)&_mainModel << " : " << (void*) sender(); |  | ||||||
|   // hier müssen wir erst das projekt aktivieren |   // hier müssen wir erst das projekt aktivieren | ||||||
|   XQItem* xqItem = static_cast<XQItem*>(item.parent()); |   XQItem* xqItem = static_cast<XQItem*>(item.parent()); | ||||||
|   onTreeViewItemClicked( *xqItem ); |   onTreeViewItemClicked( *xqItem ); | ||||||
| @@ -269,7 +267,6 @@ void XQMainWindow::onTreeViewItemChanged(const XQItem& item ) | |||||||
|     int idx = _tabWidget->currentIndex(); |     int idx = _tabWidget->currentIndex(); | ||||||
|     if(_documentStore.contains(idx) ) |     if(_documentStore.contains(idx) ) | ||||||
|     {       |     {       | ||||||
|       qDebug() << " --- Has Document and might toggle: " << item.text(); |  | ||||||
|       XQViewModel& childModel = *_documentStore[idx].viewModel; |       XQViewModel& childModel = *_documentStore[idx].viewModel; | ||||||
|       childModel.onToggleSection(item.text()); |       childModel.onToggleSection(item.text()); | ||||||
|     } |     } | ||||||
| @@ -428,44 +425,6 @@ void XQMainWindow::loadDocument( const QString& fileName, bool useQML ) | |||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| void XQMainWindow::loadDocumentQML( const QString& fileName ) |  | ||||||
| { |  | ||||||
|   // gibts die Datei? |  | ||||||
|   if( !QFile::exists( fileName) ) |  | ||||||
|     throw XQException( "no such file", fileName ); |  | ||||||
|  |  | ||||||
|   XQNodeFactory treeLoader; |  | ||||||
|   // xml daten laden |  | ||||||
|   XQNodePtr rawTree = treeLoader.load_tree( qPrintable(fileName) ); |  | ||||||
|   // versteckten root node ignorieren |  | ||||||
|   XQNodePtr contentRoot = rawTree->first_child(); |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   // 'friendly Name' ist ein Link auf ein anderes Attribute |  | ||||||
|   // das als Namen verwendet wird. |  | ||||||
|   const QString& fName = contentRoot->friendly_name(); |  | ||||||
|  |  | ||||||
|   // Ein neues Child-Model erzeugen |  | ||||||
|   XQChildModel* childModel = new XQChildModel(this); |  | ||||||
|   // die Modelstruktur anlegen |  | ||||||
|   childModel->initModel( c_ChildModelName ); |  | ||||||
|   // model inhalte laden |  | ||||||
|   childModel->addModelData( contentRoot->first_child() ); |  | ||||||
|  |  | ||||||
|   XQQuickWidget* quickChild = new XQQuickWidget(_tabWidget); |  | ||||||
|   //quickChild->setResizeMode(QQuickWidget::SizeViewToRootObject); |  | ||||||
|  |  | ||||||
|   quickChild->rootContext()->setContextProperty("xtrChildModel", childModel); |  | ||||||
|   quickChild->setSource(QUrl("qrc:/xqtreeview.qml")); |  | ||||||
|   _tabWidget->addTab( quickChild, "QML:"+fName ); |  | ||||||
|   _tabWidget->setCurrentWidget( quickChild ); |  | ||||||
|   quickChild->setResizeMode(QQuickWidget::SizeRootObjectToView); |  | ||||||
|  |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| //! speichert ein XML unter dem 'filename' | //! speichert ein XML unter dem 'filename' | ||||||
|  |  | ||||||
| void XQMainWindow::saveDocument( const QString& fileName ) | void XQMainWindow::saveDocument( const QString& fileName ) | ||||||
|   | |||||||
| @@ -64,12 +64,10 @@ public slots: | |||||||
|   // fixme implement |   // fixme implement | ||||||
|   //void showDocument( const QString& key ){} |   //void showDocument( const QString& key ){} | ||||||
|   void loadDocument( const QString& fileName, bool useQML=false ); |   void loadDocument( const QString& fileName, bool useQML=false ); | ||||||
|   void loadDocumentQML( const QString& fileName ); |  | ||||||
|   void saveDocument( const QString& fileName ); |   void saveDocument( const QString& fileName ); | ||||||
|  |  | ||||||
|   static void setupWorkingDir(); |   static void setupWorkingDir(); | ||||||
|  |  | ||||||
|  |  | ||||||
| protected: | protected: | ||||||
|  |  | ||||||
|   XQNodePtr     createDataTree( const QString& fileName ); |   XQNodePtr     createDataTree( const QString& fileName ); | ||||||
|   | |||||||
| @@ -242,7 +242,6 @@ void XQItemDelegate::drawSpinBoxStyle(QPainter* painter, const QStyleOptionViewI | |||||||
|   spinBoxOption.frame = true; |   spinBoxOption.frame = true; | ||||||
|  |  | ||||||
|   _commonStyle.drawComplexControl(QStyle::CC_SpinBox, &spinBoxOption, painter, nullptr); |   _commonStyle.drawComplexControl(QStyle::CC_SpinBox, &spinBoxOption, painter, nullptr); | ||||||
|   QRect editRect =_commonStyle.subControlRect(QStyle::CC_SpinBox, &spinBoxOption, QStyle::SC_SpinBoxEditField, nullptr); |  | ||||||
|   painter->drawText(spinBoxOption.rect, Qt::AlignCenter, item.text()); |   painter->drawText(spinBoxOption.rect, Qt::AlignCenter, item.text()); | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -42,7 +42,7 @@ void XQItemFactory::initItemFactory( const QString& modelSheetFileName ) | |||||||
|     throw XQException("modelSheet load failed. ", modelSheetFileName); |     throw XQException("modelSheet load failed. ", modelSheetFileName); | ||||||
|  |  | ||||||
|   // schritt #3: itemtype beschreibungen laden ... |   // schritt #3: itemtype beschreibungen laden ... | ||||||
|   _typesSheet =  _modelSheet->find_child_by_tag_name( "ItemTypes" ); |   _typesSheet =  _modelSheet->child( "ItemTypes" ); | ||||||
|   // ... und testen |   // ... und testen | ||||||
|   if( !_typesSheet ) |   if( !_typesSheet ) | ||||||
|     throw XQException( "initItemFactory <ItemTypes> is null" ); |     throw XQException( "initItemFactory <ItemTypes> is null" ); | ||||||
| @@ -114,7 +114,7 @@ XQItemType* XQItemFactory::findItemTypeTemplate(const QString& key ) const | |||||||
|  |  | ||||||
| XQNodePtr XQItemFactory::findModelSheet( const QString& modelName ) const | XQNodePtr XQItemFactory::findModelSheet( const QString& modelName ) const | ||||||
| { | { | ||||||
|   XQNodePtr modelSheet = _modelSheet->find_child_by_tag_name( modelName ); |   XQNodePtr modelSheet = _modelSheet->child( modelName ); | ||||||
|   if( !modelSheet ) |   if( !modelSheet ) | ||||||
|     throw XQException( "model sheet not found: ", modelName ); |     throw XQException( "model sheet not found: ", modelName ); | ||||||
|  |  | ||||||
| @@ -153,8 +153,12 @@ XQItemList XQItemFactory::makeRow(const XQNodePtr& sheetNode, const XQNodePtr& c | |||||||
|   //  value = contentNode->attributes["Capacity"]; |   //  value = contentNode->attributes["Capacity"]; | ||||||
|   // |   // | ||||||
|  |  | ||||||
|  |   // __fix! Obacht! das setzt das vorhandensein des Contents voraus! | ||||||
|  |  | ||||||
|   for( const auto& sheetEntry : sheetNode->children() ) |   for( const auto& sheetEntry : sheetNode->children() ) | ||||||
|  |   { | ||||||
|     list.append( makeItem( sheetEntry, contentNode ) ); |     list.append( makeItem( sheetEntry, contentNode ) ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   Q_ASSERT(!list.empty()); |   Q_ASSERT(!list.empty()); | ||||||
|  |  | ||||||
| @@ -165,13 +169,27 @@ XQItemList XQItemFactory::makeRow(const XQNodePtr& sheetNode, const XQNodePtr& c | |||||||
|   return list; |   return list; | ||||||
| } | } | ||||||
|  |  | ||||||
| XQItemList  XQItemFactory::makeChildRow( XQItem* parent, const XQNodePtr& sheetNode, const XQNodePtr& contentNode ) |  | ||||||
| { |  | ||||||
|   Q_UNUSED(parent); |  | ||||||
|   Q_UNUSED(sheetNode); |  | ||||||
|   Q_UNUSED(contentNode); |  | ||||||
|  |  | ||||||
|   return XQItemList(); | //! Erzeugt kind-items zu einem section-eintrag. | ||||||
|  |  | ||||||
|  | XQItemList XQItemFactory::makeChildRow( QStandardItem* parent, const XQNodePtr& sheetNode, const XQNodePtr& contentNode ) | ||||||
|  | { | ||||||
|  |   XQItemList list; | ||||||
|  |   for (const auto& contentChild : contentNode->children()) | ||||||
|  |   { | ||||||
|  |     const QString& contentKey = contentChild->tag_name(); | ||||||
|  |     // wir brauchen ein beschreibenden sheetnode für diesen content-child knoten | ||||||
|  |     if( sheetNode->has_child( contentKey )) | ||||||
|  |     { | ||||||
|  |       const XQNodePtr& sheetChild = sheetNode->child(contentKey); | ||||||
|  |       list = makeRow( sheetChild, contentChild ); | ||||||
|  |       // als kinder einfügen | ||||||
|  |       //insertRow( parent->row()+1, list ); | ||||||
|  |       parent->appendRow( list ); | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return list; | ||||||
| } | } | ||||||
|  |  | ||||||
| //! Erzeugt ein XQItem aus einer typ-beschreibung ('sheetNode') und einem daten-knoten ('contentNode'). | //! Erzeugt ein XQItem aus einer typ-beschreibung ('sheetNode') und einem daten-knoten ('contentNode'). | ||||||
|   | |||||||
| @@ -15,6 +15,7 @@ | |||||||
| #ifndef XQITEMFACTORY_H | #ifndef XQITEMFACTORY_H | ||||||
| #define XQITEMFACTORY_H | #define XQITEMFACTORY_H | ||||||
|  |  | ||||||
|  | #include <xsingleton.h> | ||||||
| #include <xqitem.h> | #include <xqitem.h> | ||||||
| #include <xqitemtype.h> | #include <xqitemtype.h> | ||||||
| #include <functional> | #include <functional> | ||||||
| @@ -32,8 +33,9 @@ public: | |||||||
|  |  | ||||||
|   XQNodePtr  findModelSheet( const QString& modelName ) const; |   XQNodePtr  findModelSheet( const QString& modelName ) const; | ||||||
|  |  | ||||||
|  |   // __fix auf nicht vorhandenen content felder prüfen! vereinheitlichen! | ||||||
|   XQItemList  makeRow( const XQNodePtr& sheetNode, const XQNodePtr& contentNode ); |   XQItemList  makeRow( const XQNodePtr& sheetNode, const XQNodePtr& contentNode ); | ||||||
|   XQItemList  makeChildRow( XQItem* parent, const XQNodePtr& sheetNode, const XQNodePtr& contentNode ); |   XQItemList  makeChildRow( QStandardItem* parent, const XQNodePtr& sheetNode, const XQNodePtr& contentNode ); | ||||||
|   XQItem*     makeSingleItem( const XQNodePtr& sheetNode, const QString& caption ); |   XQItem*     makeSingleItem( const XQNodePtr& sheetNode, const QString& caption ); | ||||||
|  |  | ||||||
|   void        setItemTypeDataFromString( XQItem& item, const QString& roleKey, const QString& source ) const; |   void        setItemTypeDataFromString( XQItem& item, const QString& roleKey, const QString& source ) const; | ||||||
|   | |||||||
							
								
								
									
										129
									
								
								src/main.cpp
									
									
									
									
									
								
							
							
						
						
									
										129
									
								
								src/main.cpp
									
									
									
									
									
								
							| @@ -17,6 +17,7 @@ | |||||||
| #include <QMetaType> | #include <QMetaType> | ||||||
| #include <QGuiApplication> | #include <QGuiApplication> | ||||||
| #include <QQmlApplicationEngine> | #include <QQmlApplicationEngine> | ||||||
|  | #include <QQuickStyle> | ||||||
| #include <QUrl> | #include <QUrl> | ||||||
| #include <QQmlContext> | #include <QQmlContext> | ||||||
| #include <QStyleFactory> | #include <QStyleFactory> | ||||||
| @@ -27,140 +28,20 @@ | |||||||
| #include <xqmainwindow.h> | #include <xqmainwindow.h> | ||||||
|  |  | ||||||
|  |  | ||||||
| /* |  | ||||||
|  |  | ||||||
|   TODO: |  | ||||||
|  |  | ||||||
|   - kann ich die Enums auto generieren? |  | ||||||
|       - entries einfach abzählen, |  | ||||||
|       - einen enum als type |  | ||||||
|       - auf diesen type drauf-casten, siehe qtglobal.h |  | ||||||
|  |  | ||||||
|   - FIX! in reality, we have nested types, also. |  | ||||||
|   - done: reference style |  | ||||||
|   - done: shared_ptr |  | ||||||
|  |  | ||||||
|   - try QML |  | ||||||
|   - try 'model->readMore' |  | ||||||
|  |  | ||||||
| doc: |  | ||||||
|  - vorhandenes xnode konzept soll am qt angebunden werden |  | ||||||
|  - datensparsam: flyweight pattern & pointer auf orig |  | ||||||
|  |  | ||||||
| who is who: |  | ||||||
|  |  | ||||||
| - item |  | ||||||
| -itemtype |  | ||||||
|  - factory |  | ||||||
|  - model |  | ||||||
|  -  section |  | ||||||
|  |  | ||||||
| */ |  | ||||||
|  |  | ||||||
| XQChildModel* createChildModel() |  | ||||||
| { |  | ||||||
|   XQChildModel* myModel = new XQChildModel(); |  | ||||||
|  |  | ||||||
|   myModel->initModel( c_ChildModelName ); |  | ||||||
|  |  | ||||||
|   XQNodeFactory treeLoader; |  | ||||||
|   // xml daten laden |  | ||||||
|   XQNodePtr rawTree = treeLoader.load_tree( qPrintable(c_DocumentFileName1) ); |  | ||||||
|   // versteckten root node ignorieren |  | ||||||
|   XQNodePtr contentRoot = rawTree->first_child(); |  | ||||||
|   myModel->addModelData( contentRoot->first_child() ); |  | ||||||
|  |  | ||||||
|   return myModel; |  | ||||||
|  |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* |  | ||||||
|  |  | ||||||
|  |  | ||||||
| CONFIG += precompile_header |  | ||||||
| PRECOMPILED_HEADER = pch.h |  | ||||||
|  |  | ||||||
| // pch.h – Precompiled Header für Qt-Projekt |  | ||||||
|  |  | ||||||
| Stelle sicher, dass pch.h im Projektordner liegt und in den .cpp-Dateien eingebunden wird: |  | ||||||
|  |  | ||||||
| C++ |  | ||||||
| #include "pch.h" |  | ||||||
|  |  | ||||||
| #ifndef PCH_H |  | ||||||
| #define PCH_H |  | ||||||
|  |  | ||||||
| // Qt Core |  | ||||||
| #include <QtCore/QCoreApplication> |  | ||||||
| #include <QtCore/QDebug> |  | ||||||
| #include <QtCore/QObject> |  | ||||||
| #include <QtCore/QTimer> |  | ||||||
| #include <QtCore/QDateTime> |  | ||||||
|  |  | ||||||
| // Qt GUI |  | ||||||
| #include <QtGui/QGuiApplication> |  | ||||||
| #include <QtGui/QIcon> |  | ||||||
| #include <QtGui/QPixmap> |  | ||||||
|  |  | ||||||
| // Qt Widgets (falls verwendet) |  | ||||||
| #include <QtWidgets/QApplication> |  | ||||||
| #include <QtWidgets/QMainWindow> |  | ||||||
| #include <QtWidgets/QPushButton> |  | ||||||
| #include <QtWidgets/QLabel> |  | ||||||
|  |  | ||||||
| // STL |  | ||||||
| #include <vector> |  | ||||||
| #include <string> |  | ||||||
| #include <map> |  | ||||||
| #include <memory> |  | ||||||
|  |  | ||||||
| #endif // PCH_H |  | ||||||
|  |  | ||||||
| */ |  | ||||||
|  |  | ||||||
| using namespace Qt::Literals::StringLiterals; | using namespace Qt::Literals::StringLiterals; | ||||||
|  |  | ||||||
| int main(int argc, char *argv[]) | int main(int argc, char *argv[]) | ||||||
| { | { | ||||||
|  |  | ||||||
|   QApplication app(argc, argv); |   QApplication app(argc, argv); | ||||||
|   //qDebug() << QStyleFactory::keys(); |  | ||||||
|   //QApplication::setStyle("fusion"); |   // hat am wenigsten darstellungfehler (alternative: fusion) | ||||||
|   //QApplication::setStyle("windowsvista"); |   QQuickStyle::setStyle("Imagine"); | ||||||
|   //QApplication::setStyle("windows"); |  | ||||||
|  |  | ||||||
|   XQMainWindow window; |   XQMainWindow window; | ||||||
|   window.show(); |   window.show(); | ||||||
|  |  | ||||||
|   /* |  | ||||||
|   QApplication app(argc, argv); |  | ||||||
|  |  | ||||||
|   XQMainWindow::setupWorkingDir(); |  | ||||||
|   XQItemFactory::instance().initItemFactory( c_ModelSheetFileName ); |  | ||||||
|  |  | ||||||
|   XQChildModel* myModel = createChildModel(); |  | ||||||
|   qmlRegisterType< XQChildModel>("MyApp.Models", 1, 0, "MyChildModel"); |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   QQmlApplicationEngine engine; |  | ||||||
|  |  | ||||||
|     engine.rootContext()->setContextProperty("myChildModel", myModel); |  | ||||||
|  |  | ||||||
|     QObject::connect( |  | ||||||
|       &engine, |  | ||||||
|       &QQmlApplicationEngine::objectCreationFailed, |  | ||||||
|       &app, |  | ||||||
|       []() { QCoreApplication::exit(-1); }, |  | ||||||
|       Qt::QueuedConnection); |  | ||||||
|  |  | ||||||
|     qDebug() << " fitz!"; |  | ||||||
|  |  | ||||||
|     engine.load( QUrl(QStringLiteral("qrc:/dummyview.qml")) ); |  | ||||||
|     //engine.load( QUrl(QStringLiteral("qrc:/xqtableview.qml")) ); |  | ||||||
|  |  | ||||||
|     qDebug() << " hhakl!"; |  | ||||||
|  |  | ||||||
| */ |  | ||||||
|   return app.exec(); |   return app.exec(); | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -16,8 +16,6 @@ | |||||||
| #include <xqitem.h> | #include <xqitem.h> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| //! hilfsfunktion: gibt diesen teilbaum rekursiv aus | //! hilfsfunktion: gibt diesen teilbaum rekursiv aus | ||||||
|  |  | ||||||
| void inspect( const XQNodePtr& node, int indent ) | void inspect( const XQNodePtr& node, int indent ) | ||||||
|   | |||||||
| @@ -16,8 +16,6 @@ | |||||||
| #define XQNODE_H | #define XQNODE_H | ||||||
|  |  | ||||||
| #include <iostream> | #include <iostream> | ||||||
| #include <memory> |  | ||||||
|  |  | ||||||
| #include <QDebug> | #include <QDebug> | ||||||
| #include <QModelIndex> | #include <QModelIndex> | ||||||
|  |  | ||||||
| @@ -29,12 +27,12 @@ | |||||||
| std::ostream& operator<<(std::ostream& os, const QString& obj); | std::ostream& operator<<(std::ostream& os, const QString& obj); | ||||||
|  |  | ||||||
| //! raw node | //! raw node | ||||||
| using XQNode = znode::zbasic_node<QString>; | using XQNode = znode::zbasic_node<QString,znode::zpayload<QString>>; | ||||||
| //! default shared node | //! default shared node | ||||||
| using XQNodePtr = std::shared_ptr<znode::zbasic_node<QString>>; | using XQNodePtr = std::shared_ptr<znode::zbasic_node<QString,znode::zpayload<QString>>>; | ||||||
|  |  | ||||||
| //! die node factory | //! die node factory | ||||||
| using XQNodeFactory = znode::znode_factory<QString>; | using XQNodeFactory = znode::znode_factory<QString,znode::zpayload<QString>>; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -60,7 +60,7 @@ XQNodePtr XQModelSection::sectionRootNode() const | |||||||
|  |  | ||||||
| XQNodePtr XQModelSection::sheetRootNode() const | XQNodePtr XQModelSection::sheetRootNode() const | ||||||
| { | { | ||||||
|   return _sectionSheetRootNode->find_child_by_tag_name( c_ModelSheet ); |   return _sectionSheetRootNode->child( c_ModelSheet ); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -139,7 +139,7 @@ void XQViewModel::initModel(const QString& modelName) | |||||||
|   for( auto& sectionNode : modelSheet->children() ) |   for( auto& sectionNode : modelSheet->children() ) | ||||||
|   { |   { | ||||||
|     // #2: (optionalen?) header erzeugen |     // #2: (optionalen?) header erzeugen | ||||||
|     const XQNodePtr header = sectionNode->find_child_by_tag_name( c_Header ); |     const XQNodePtr header = sectionNode->child( c_Header ); | ||||||
|     if( header ) |     if( header ) | ||||||
|     { |     { | ||||||
|       XQItemList list = _itemFactory.makeRow( header, nullptr ); |       XQItemList list = _itemFactory.makeRow( header, nullptr ); | ||||||
| @@ -166,10 +166,10 @@ void XQViewModel::addSection(const XQItemList& list, const XQNodePtr& sheetNode | |||||||
|   if( !sheetNode->has_attribute( c_ContentType) ) |   if( !sheetNode->has_attribute( c_ContentType) ) | ||||||
|     throw XQException( "section list: Section node needs attribute 'ContentType'!"); |     throw XQException( "section list: Section node needs attribute 'ContentType'!"); | ||||||
|  |  | ||||||
|   // 5. das erzeugt dann auch valide indices |   // 4. das erzeugt dann auch valide indices | ||||||
|   appendRow(list); |   appendRow(list); | ||||||
|  |  | ||||||
|   // 6. jetzt können wir auch die section erzeugen |   // 5. jetzt können wir auch die section erzeugen | ||||||
|   const XQModelSection& section = _sections.createSection( list[0]->index(), sheetNode ); |   const XQModelSection& section = _sections.createSection( list[0]->index(), sheetNode ); | ||||||
|  |  | ||||||
|   // ... und es der welt mitteilen. |   // ... und es der welt mitteilen. | ||||||
| @@ -182,8 +182,6 @@ void XQViewModel::addSection(const XQItemList& list, const XQNodePtr& sheetNode | |||||||
|  |  | ||||||
| void XQViewModel::onToggleSection(const QString& sectionKey ) | void XQViewModel::onToggleSection(const QString& sectionKey ) | ||||||
| { | { | ||||||
|   qDebug() <<  " --- Model: " << this->objectName() << " should toggle: " << sectionKey << ": " << _sections.hasValidSection( sectionKey ); |  | ||||||
|   _sections.dump(); |  | ||||||
|   if(_sections.hasValidSection( sectionKey ) ) |   if(_sections.hasValidSection( sectionKey ) ) | ||||||
|     toggleSection( _sections.sectionByKey(sectionKey) ); |     toggleSection( _sections.sectionByKey(sectionKey) ); | ||||||
| } | } | ||||||
| @@ -329,25 +327,12 @@ void XQViewModel::onCommandUndo( const XQCommand& command ) | |||||||
|  |  | ||||||
| void XQViewModel::cmdCut( const XQCommand& command ) | void XQViewModel::cmdCut( const XQCommand& command ) | ||||||
| { | { | ||||||
|  |  | ||||||
|   int itmPos  = command.first().itemPos; |  | ||||||
|   const XQModelSection& section = _sections.sectionByRow( itmPos ); |  | ||||||
|   qDebug()  << " --- HEADSHOT I: " << itmPos << "->" << section.contentType(); |  | ||||||
|  |  | ||||||
|   // wir gehen rückwärts über alle gemerkten knoten ... |   // wir gehen rückwärts über alle gemerkten knoten ... | ||||||
|   for (auto it = command.rbegin(); it != command.rend(); ++it) |   for (auto it = command.rbegin(); it != command.rend(); ++it) | ||||||
|   { |   { | ||||||
|  |  | ||||||
|  |  | ||||||
|    // ... holen das erste item, das auch den content node enthält |    // ... holen das erste item, das auch den content node enthält | ||||||
|     //const XQNodeBackup& entry = *it; |  | ||||||
|     // jetzt löschen, dabei wird die parent-verbindung entfernt |     // jetzt löschen, dabei wird die parent-verbindung entfernt | ||||||
|     const XQNodeBackup& entry = *it; |     const XQNodeBackup& entry = *it; | ||||||
|  |  | ||||||
|     XQItem& firstItem = xqFirstItem( (*it).itemPos ); |  | ||||||
|     //qDebug() << " --- Cut: "  << firstItem.text() << " " << firstItem.row() << " id#" << entry.contentNode->_id; |  | ||||||
|     qDebug() << " ---- command CUT: itemPos: " << entry.itemPos << "  nodePos: "<< entry.nodePos << " is " << entry.contentNode->friendly_name(); |  | ||||||
|  |  | ||||||
|     entry.contentNode->unlink_self(); |     entry.contentNode->unlink_self(); | ||||||
|     removeRow(entry.itemPos ); |     removeRow(entry.itemPos ); | ||||||
|   } |   } | ||||||
| @@ -358,14 +343,11 @@ void XQViewModel::cmdCut( const XQCommand& command ) | |||||||
|  |  | ||||||
| void XQViewModel::cmdCutUndo( const XQCommand& command ) | void XQViewModel::cmdCutUndo( const XQCommand& command ) | ||||||
| { | { | ||||||
|  |  | ||||||
|   // die anfangsposition |   // die anfangsposition | ||||||
|   int itmPos  = command.first().itemPos; |   int itmPos  = command.first().itemPos; | ||||||
|   // die 'zuständige' section rausfinden |   // die 'zuständige' section rausfinden | ||||||
|   const XQModelSection& section = _sections.sectionByRow( itmPos ); |   const XQModelSection& section = _sections.sectionByRow( itmPos ); | ||||||
|  |  | ||||||
|   qDebug()  << " --- HEADSHOT II: " << itmPos << "->" << section.contentType(); |  | ||||||
|  |  | ||||||
|   // über alle einträge ... |   // über alle einträge ... | ||||||
|   for (auto& entry : command ) |   for (auto& entry : command ) | ||||||
|   { |   { | ||||||
| @@ -374,12 +356,6 @@ void XQViewModel::cmdCutUndo( const XQCommand& command ) | |||||||
|     savedNode->add_me_at( entry.nodePos, _contentRoot ); |     savedNode->add_me_at( entry.nodePos, _contentRoot ); | ||||||
|     XQItemList list = _itemFactory.makeRow( section.sheetRootNode(), savedNode ); |     XQItemList list = _itemFactory.makeRow( section.sheetRootNode(), savedNode ); | ||||||
|     insertRow( entry.itemPos, list ); |     insertRow( entry.itemPos, list ); | ||||||
|  |  | ||||||
|     XQItem& firstItem = *((XQItem*)list[0]); |  | ||||||
|  |  | ||||||
|     qDebug() << " ---- command cut UNDO2: itemPos: " << entry.itemPos << "  nodePos: "<< entry.nodePos << " is " << entry.contentNode->friendly_name(); |  | ||||||
|     qDebug() << " --- Cut Undo: "  << firstItem.text() << " " << firstItem.row() << " id#" << entry.contentNode->_id << " count: " << entry.contentNode.use_count(); |  | ||||||
|  |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -421,8 +397,6 @@ void XQViewModel::cmdPaste( const XQCommand& command ) | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   // unsere änderungen merken fürs 'undo' |   // unsere änderungen merken fürs 'undo' | ||||||
|  |  | ||||||
|   /// fix_xx |  | ||||||
|   const_cast<XQCommand&>(command).saveNodes( selectionModel->selectedRows() ); |   const_cast<XQCommand&>(command).saveNodes( selectionModel->selectedRows() ); | ||||||
|  |  | ||||||
| } | } | ||||||
| @@ -432,14 +406,11 @@ void XQViewModel::cmdPaste( const XQCommand& command ) | |||||||
|  |  | ||||||
| void XQViewModel::cmdPasteUndo( const XQCommand& command ) | void XQViewModel::cmdPasteUndo( const XQCommand& command ) | ||||||
| { | { | ||||||
|   command.dumpList("Paste UNDO"); |  | ||||||
|   // wir gehen rückwärts über alle markieren knoten ... |   // wir gehen rückwärts über alle markieren knoten ... | ||||||
|   for (auto it = command.rbegin(); it != command.rend(); ++it) |   for (auto it = command.rbegin(); it != command.rend(); ++it) | ||||||
|   { |   { | ||||||
|     // ... holen das erste item, das auch den content node enthält |     // ... holen das erste item, das auch den content node enthält | ||||||
|     const XQNodeBackup& entry = *it; |     const XQNodeBackup& entry = *it; | ||||||
|     XQItem& firstItem = xqFirstItem( (*it).itemPos ); |  | ||||||
|     qDebug() << " --- Cut: "  << firstItem.text() << " " << firstItem.row(); |  | ||||||
|     // jetzt löschen |     // jetzt löschen | ||||||
|     entry.contentNode->unlink_self(); |     entry.contentNode->unlink_self(); | ||||||
|     removeRow(entry.itemPos ); |     removeRow(entry.itemPos ); | ||||||
| @@ -458,8 +429,6 @@ void XQViewModel::cmdDelete( const XQCommand& command ) | |||||||
|   { |   { | ||||||
|     // ... holen das erste item, das auch den content node enthält |     // ... holen das erste item, das auch den content node enthält | ||||||
|     const XQNodeBackup& entry = *it; |     const XQNodeBackup& entry = *it; | ||||||
|     XQItem& firstItem = xqFirstItem( (*it).itemPos ); |  | ||||||
|     qDebug() << " --- delete: "  << firstItem.text() << " " << firstItem.row(); |  | ||||||
|     // jetzt löschen |     // jetzt löschen | ||||||
|     entry.contentNode->unlink_self(); |     entry.contentNode->unlink_self(); | ||||||
|     removeRow(entry.itemPos ); |     removeRow(entry.itemPos ); | ||||||
| @@ -470,10 +439,6 @@ void XQViewModel::cmdDelete( const XQCommand& command ) | |||||||
|  |  | ||||||
| void XQViewModel::cmdDeleteUndo( const XQCommand& command ) | void XQViewModel::cmdDeleteUndo( const XQCommand& command ) | ||||||
| { | { | ||||||
|   for (const auto& entry : command) |  | ||||||
|   { |  | ||||||
|     qDebug() << " --- delete UNDo: " << entry.contentNode->friendly_name(); |  | ||||||
|   } |  | ||||||
|   cmdCutUndo(command); |   cmdCutUndo(command); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -491,8 +456,6 @@ void XQViewModel::cmdNew( const XQCommand& command ) | |||||||
|   XQNodePtr newNode = XQNode::make_node( node->tag_name(), node->tag_value() ); |   XQNodePtr newNode = XQNode::make_node( node->tag_name(), node->tag_value() ); | ||||||
|   // store node in node->parent() |   // store node in node->parent() | ||||||
|   newNode->add_me_at( node->own_pos(), node->parent() ); |   newNode->add_me_at( node->own_pos(), node->parent() ); | ||||||
|  |  | ||||||
|  |  | ||||||
|   //... |   //... | ||||||
|   const XQModelSection& section = _sections.sectionByRow( origin.row() ); |   const XQModelSection& section = _sections.sectionByRow( origin.row() ); | ||||||
|  |  | ||||||
| @@ -531,7 +494,6 @@ void XQViewModel::cmdToggleSection( const XQCommand& command ) | |||||||
|   // einen doppelten Aufruf und wir sind dann wieder im Anfangszustand. |   // einen doppelten Aufruf und wir sind dann wieder im Anfangszustand. | ||||||
|  |  | ||||||
|   //toggleSection( section ); |   //toggleSection( section ); | ||||||
|  |  | ||||||
|   emit sectionToggled(section); |   emit sectionToggled(section); | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -17,14 +17,11 @@ | |||||||
|  |  | ||||||
|  |  | ||||||
| #include <vector> | #include <vector> | ||||||
| #include <string> |  | ||||||
| #include <memory> | #include <memory> | ||||||
| //#include <typeinfo> |  | ||||||
|  |  | ||||||
| #include <QString> | #include <QString> | ||||||
| #include <QDebug> | #include <QDebug> | ||||||
| #include <xqmaptor.h> | #include <xqmaptor.h> | ||||||
| #include <xsingleton.h> |  | ||||||
|  |  | ||||||
| #include <znode_iterator.h> | #include <znode_iterator.h> | ||||||
| //#include <znode_vector.h> | //#include <znode_vector.h> | ||||||
| @@ -41,37 +38,21 @@ namespace znode | |||||||
|  |  | ||||||
|    */ |    */ | ||||||
|  |  | ||||||
|  |     //! tree-klasse, besonderheit: der nutzlast & string-type sind templated. | ||||||
|     // forward declaration of base class zbasic_node ... |     template<typename str_t, typename payload_t> | ||||||
|     //template<typename str_t> |     class zbasic_node : public zid, public payload_t, public std::enable_shared_from_this<zbasic_node<str_t,payload_t>> | ||||||
|     //class zbasic_node; |  | ||||||
|  |  | ||||||
|     // ... used here for conveniance typedef |  | ||||||
|     //template<typename str_t> |  | ||||||
|     //using zshared_node = std::shared_ptr<zbasic_node<str_t>>; |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     //! einfache tree-klasse, besonderheit: der nutzlast-string-type ist templated. |  | ||||||
|     template<typename str_t> |  | ||||||
|     //class zbasic_node : public zid, public zpayload<str_t>, public std::enable_shared_from_this<zbasic_node<str_t>> |  | ||||||
|     class zbasic_node : public zid, public zpayload<str_t>, public std::enable_shared_from_this<zbasic_node<str_t>> |  | ||||||
|     { |     { | ||||||
|  |  | ||||||
|     public: |     public: | ||||||
|  |  | ||||||
|       using str_cref     = const str_t&; |  | ||||||
|       using str_list     = std::vector<str_t>; |  | ||||||
|  |  | ||||||
|       //using zshared_node = std::shared_ptr<zbasic_node>; |  | ||||||
|       //using znode_list   = znode_vector<zbasic_node>; |  | ||||||
|       //using znode_list   = znode_vector<zshared_node<str_t>>; |  | ||||||
|       //using znode_list   = std::vector<zshared_node<str_t>>; |  | ||||||
|  |  | ||||||
|       using zweak_node   = std::weak_ptr<zbasic_node>; |       using zweak_node   = std::weak_ptr<zbasic_node>; | ||||||
|       using zshared_node = std::shared_ptr<zbasic_node>; |       using zshared_node = std::shared_ptr<zbasic_node>; | ||||||
|       using znode_list   = std::vector<zshared_node>; |       using znode_list   = std::vector<zshared_node>; | ||||||
|       using zshared_cref = const std::shared_ptr<zbasic_node>&; |       using zshared_cref = const std::shared_ptr<zbasic_node>&; | ||||||
|  |  | ||||||
|  |       using str_cref     = const str_t&; | ||||||
|  |       using str_list     = std::vector<str_t>; | ||||||
|  |  | ||||||
|       using ziterator    = znode_iterator<zbasic_node>; |       using ziterator    = znode_iterator<zbasic_node>; | ||||||
|  |  | ||||||
|     protected: |     protected: | ||||||
| @@ -109,14 +90,14 @@ namespace znode | |||||||
|  |  | ||||||
|       //! konstruktor mit tag_name und optionalem elternknoten |       //! konstruktor mit tag_name und optionalem elternknoten | ||||||
|       zbasic_node( str_cref tag_name, zshared_cref parent = nullptr ) |       zbasic_node( str_cref tag_name, zshared_cref parent = nullptr ) | ||||||
|         : zpayload<str_t>{tag_name}, _parent{parent} |         : payload_t{tag_name}, _parent{parent} | ||||||
|       { |       { | ||||||
|  |  | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       //! konstruktor mit tag_name, value und optionalem elternknoten |       //! konstruktor mit tag_name, value und optionalem elternknoten | ||||||
|       zbasic_node( str_cref tag_name, str_cref value, zshared_cref parent = nullptr ) |       zbasic_node( str_cref tag_name, str_cref value, zshared_cref parent = nullptr ) | ||||||
|          : zpayload<str_t>{tag_name,value}, _parent{parent} |          : payload_t{tag_name,value}, _parent{parent} | ||||||
|       { |       { | ||||||
|  |  | ||||||
|       } |       } | ||||||
| @@ -206,6 +187,17 @@ namespace znode | |||||||
|         return !children().empty(); |         return !children().empty(); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       bool has_child(int idx) const | ||||||
|  |       { | ||||||
|  |         return (idx>-1 && idx<children().size()); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       //! testet, ob kinder vorhanden sind. | ||||||
|  |       bool has_child(str_cref tagname ) const | ||||||
|  |       { | ||||||
|  |         return child(tagname ) != nullptr; | ||||||
|  |       } | ||||||
|  |  | ||||||
|       //! gibt das erste kind zurück |       //! gibt das erste kind zurück | ||||||
|       zshared_node first_child() |       zshared_node first_child() | ||||||
|       { |       { | ||||||
| @@ -229,6 +221,17 @@ namespace znode | |||||||
|         return nullptr; |         return nullptr; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       // | ||||||
|  |       zshared_node child(str_cref tagname ) const | ||||||
|  |       { | ||||||
|  |         for( auto child : _children ) | ||||||
|  |         { | ||||||
|  |           if( child->tag_name() == tagname ) | ||||||
|  |             return child; | ||||||
|  |         } | ||||||
|  |         return nullptr; | ||||||
|  |       } | ||||||
|  |  | ||||||
|       //! hängt einen knoten an meine kinderliste an. |       //! hängt einen knoten an meine kinderliste an. | ||||||
|       int add_child( const zshared_node& node ) |       int add_child( const zshared_node& node ) | ||||||
|       { |       { | ||||||
| @@ -313,7 +316,7 @@ namespace znode | |||||||
|  |  | ||||||
|       //! findet den ersten kind-knoten mit dem attribut 'attrkey' welches den |       //! findet den ersten kind-knoten mit dem attribut 'attrkey' welches den | ||||||
|       //! wert 'attrvalue' hat. |       //! wert 'attrvalue' hat. | ||||||
|       zshared_node find_child_by_attribute(str_cref attrkey, str_cref attrvalue ) |       zshared_node find_child_by_attribute(str_cref attrkey, str_cref attrvalue ) const | ||||||
|       { |       { | ||||||
|         for( auto child : _children ) |         for( auto child : _children ) | ||||||
|         { |         { | ||||||
| @@ -323,18 +326,9 @@ namespace znode | |||||||
|         return nullptr; |         return nullptr; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       // |  | ||||||
|       zshared_node find_child_by_tag_name(str_cref tagname ) |  | ||||||
|       { |  | ||||||
|         for( auto child : _children ) |  | ||||||
|         { |  | ||||||
|           if( child->tag_name() == tagname ) |  | ||||||
|             return child; |  | ||||||
|         } |  | ||||||
|         return nullptr; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       zshared_node find_child_by_id( int id ) |  | ||||||
|  |       zshared_node find_child_by_id( int id ) const | ||||||
|       { |       { | ||||||
|         for (auto child : _children ) |         for (auto child : _children ) | ||||||
|         { |         { | ||||||
|   | |||||||
| @@ -40,15 +40,14 @@ model: muss ich wirklich jeden attibute node einzeln angeben? | |||||||
| */ | */ | ||||||
| namespace znode | namespace znode | ||||||
| { | { | ||||||
|   template<typename str_t> |   template<typename str_t,typename payload_t> | ||||||
|   class znode_factory |   class znode_factory | ||||||
|   { |   { | ||||||
|  |  | ||||||
|   public: |   public: | ||||||
|  |  | ||||||
|     using str_cref     = const str_t&; |     using str_cref     = const str_t&; | ||||||
|  |     using zshared_node = std::shared_ptr< zbasic_node<str_t,payload_t> >; | ||||||
|     using zshared_node = std::shared_ptr<zbasic_node<str_t>>; |  | ||||||
|  |  | ||||||
|     znode_factory() = default; |     znode_factory() = default; | ||||||
|     virtual ~znode_factory() = default; |     virtual ~znode_factory() = default; | ||||||
| @@ -70,7 +69,7 @@ namespace znode | |||||||
|       //tree_document.traverse(parser); |       //tree_document.traverse(parser); | ||||||
|  |  | ||||||
|       //zbasic_node<str_t>* root_node = new zbasic_node<str_t>*("root!"); |       //zbasic_node<str_t>* root_node = new zbasic_node<str_t>*("root!"); | ||||||
|       zshared_node root_node = zbasic_node<str_t>::make_node("root!"); |       zshared_node root_node = zbasic_node<str_t, payload_t>::make_node("root!"); | ||||||
|       //T root_node = T::make_node( "root!" ); |       //T root_node = T::make_node( "root!" ); | ||||||
|  |  | ||||||
|       // prepare root == model !? |       // prepare root == model !? | ||||||
| @@ -100,7 +99,7 @@ namespace znode | |||||||
|       //parent->add_child( new_node ); |       //parent->add_child( new_node ); | ||||||
|  |  | ||||||
|       //zbasic_node<str_t>* new_node = new zbasic_node<str_t>( node.name(), node.child_value(), parent ); |       //zbasic_node<str_t>* new_node = new zbasic_node<str_t>( node.name(), node.child_value(), parent ); | ||||||
|       zshared_node new_node = zbasic_node<str_t>::make_node( xml_node.name(), xml_node.child_value() ); |       zshared_node new_node = zbasic_node<str_t, payload_t>::make_node( xml_node.name(), xml_node.child_value() ); | ||||||
|       parent->add_child( new_node ); |       parent->add_child( new_node ); | ||||||
|  |  | ||||||
|       if( !xml_node.attributes().empty() ) |       if( !xml_node.attributes().empty() ) | ||||||
|   | |||||||
| @@ -630,7 +630,7 @@ namespace pugi | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// Find child node using predicate. Returns first child for which predicate returned true. | 		// Find child node using predicate. Returns first child for which predicate returned true. | ||||||
| 		template <typename Predicate> xml_node find_child_by_tag_name(Predicate pred) const | 		template <typename Predicate> xml_node child(Predicate pred) const | ||||||
| 		{ | 		{ | ||||||
| 			if (!_root) return xml_node(); | 			if (!_root) return xml_node(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| QT       += core gui widgets quick quickwidgets | QT       += core gui widgets quick quickwidgets quickcontrols2 | ||||||
| # widgets-private | # widgets-private | ||||||
|  |  | ||||||
| CONFIG += c++20 qmltypes | CONFIG += c++20 qmltypes | ||||||
|   | |||||||
| @@ -16,7 +16,8 @@ | |||||||
| 		<Inverter InverterID="#4 D12K 04" FriendlyName="@InverterName" InverterName="04 HM600 S2 TMax" Manufacturer="Deye"  MaxPowerInput="8000" MaxPowerInputChoice="6000;8000;12000" MaxPowerOutput="600" NumStrings="2" Weight="28"/>				 | 		<Inverter InverterID="#4 D12K 04" FriendlyName="@InverterName" InverterName="04 HM600 S2 TMax" Manufacturer="Deye"  MaxPowerInput="8000" MaxPowerInputChoice="6000;8000;12000" MaxPowerOutput="600" NumStrings="2" Weight="28"/>				 | ||||||
| 		<Battery BatteryID="#1 BYD 01" FriendlyName="@BatteryName" BatteryName="01 BYD T01 Stackable" Manufacturer="BYD" Capacity="4500" Yield="90" MaxCurrent="120" MaxVolt="48">		 | 		<Battery BatteryID="#1 BYD 01" FriendlyName="@BatteryName" BatteryName="01 BYD T01 Stackable" Manufacturer="BYD" Capacity="4500" Yield="90" MaxCurrent="120" MaxVolt="48">		 | ||||||
| 			<Images FrontView="image.png" PackageView="package.png" InstalledView="installed.png"/> | 			<Images FrontView="image.png" PackageView="package.png" InstalledView="installed.png"/> | ||||||
| 			<Documents Manual="manual.docx" Certificate="certificate.pdf"/>			 | 			<Documents Manual="manual.docx" Installation="installation.pdf"/>		 | ||||||
|  | 			<Certificates Certificate="certificate.pdf" Conformity="conformity.pdf"/> | ||||||
| 		</Battery> | 		</Battery> | ||||||
| 		<Battery BatteryID="#2 BYD 02" FriendlyName="@BatteryName" BatteryName="02 BYD T02 Stackable" Manufacturer="BYD" Capacity="9000" Yield="94" MaxCurrent="120" MaxVolt="48"/>	 | 		<Battery BatteryID="#2 BYD 02" FriendlyName="@BatteryName" BatteryName="02 BYD T02 Stackable" Manufacturer="BYD" Capacity="9000" Yield="94" MaxCurrent="120" MaxVolt="48"/>	 | ||||||
| 		<Battery BatteryID="#3 BYD 03" FriendlyName="@BatteryName" BatteryName="03 BYD T01 Stackable" Manufacturer="BYD" Capacity="4500" Yield="86" MaxCurrent="120" MaxVolt="48"/>		 | 		<Battery BatteryID="#3 BYD 03" FriendlyName="@BatteryName" BatteryName="03 BYD T01 Stackable" Manufacturer="BYD" Capacity="4500" Yield="86" MaxCurrent="120" MaxVolt="48"/>		 | ||||||
|   | |||||||
| @@ -1,12 +1,7 @@ | |||||||
| <?xml version='1.0' encoding='UTF-8'?> | <?xml version='1.0' encoding='UTF-8'?> | ||||||
|  |  | ||||||
|  |  | ||||||
| <Project ProjectID="HA02" FriendlyName="@ProjectName" ProjectName="Gerbrunn Ost" Established="2006" WattPeak="9840"  ContentType="planned"> | <Project ProjectID="HA02" FriendlyName="@ProjectName" ProjectName="Gerbrunn Ost" Established="2006" WattPeak="9840"  ContentType="planned"> | ||||||
| 	<Components>	 | 	<Components>	 | ||||||
| 		<Panel PanelID="Jingli 01" FriendlyName="@PanelName" PanelName="Jingli 01 Solar T62B" Manufacturer="Jingli Solar" WattPeak="620" Height="2,70" Width="1,10" Weight="12" MaxVolt="67" MaxAmpere="11">			 | 		<Panel PanelID="Jingli 01" FriendlyName="@PanelName" PanelName="Jingli 01 Solar T62B" Manufacturer="Jingli Solar" WattPeak="620" Height="2,70" Width="1,10" Weight="12" MaxVolt="67" MaxAmpere="11">			 | ||||||
| 			<AdditionalData DataItem="Image"       DataValue="image.png"/> |  | ||||||
| 			<AdditionalData DataItem="Manual"      DataValue="manual.docx"/> |  | ||||||
| 			<AdditionalData DataItem="Certificate" DataValue="certificate.pdf"/> |  | ||||||
| 		</Panel> | 		</Panel> | ||||||
| 		<Panel PanelID="Jingli 02" FriendlyName="@PanelName" PanelName="Jingli 02 Solar X58C" Manufacturer="Jingli Solar" WattPeak="440" Height="1,70" Width="1,10" Weight="12" MaxVolt="42" MaxAmpere="11"/>			 | 		<Panel PanelID="Jingli 02" FriendlyName="@PanelName" PanelName="Jingli 02 Solar X58C" Manufacturer="Jingli Solar" WattPeak="440" Height="1,70" Width="1,10" Weight="12" MaxVolt="42" MaxAmpere="11"/>			 | ||||||
| 		<Panel PanelID="Jingli 03" FriendlyName="@PanelName" PanelName="Jingli 03 Solar T62B" Manufacturer="Jingli Solar" WattPeak="620" Height="2,70" Width="1,10" Weight="12" MaxVolt="67" MaxAmpere="11"/>			 | 		<Panel PanelID="Jingli 03" FriendlyName="@PanelName" PanelName="Jingli 03 Solar T62B" Manufacturer="Jingli Solar" WattPeak="620" Height="2,70" Width="1,10" Weight="12" MaxVolt="67" MaxAmpere="11"/>			 | ||||||
| @@ -25,7 +20,6 @@ | |||||||
| 		<Battery BatteryID="GroWatt 05 G2K" FriendlyName="@BatteryName" BatteryName="05 BYD T01 Stackable" Manufacturer="PylonTech" Capacity="4500" Yield="46" MaxCurrent="120" MaxVolt="48"/>		 | 		<Battery BatteryID="GroWatt 05 G2K" FriendlyName="@BatteryName" BatteryName="05 BYD T01 Stackable" Manufacturer="PylonTech" Capacity="4500" Yield="46" MaxCurrent="120" MaxVolt="48"/>		 | ||||||
| 		<Battery BatteryID="GroWatt 06 G4K" FriendlyName="@BatteryName" BatteryName="06 BYD T02 Stackable" Manufacturer="PylonTech" Capacity="9000"  Yield="94" MaxCurrent="120" MaxVolt="48">										 | 		<Battery BatteryID="GroWatt 06 G4K" FriendlyName="@BatteryName" BatteryName="06 BYD T02 Stackable" Manufacturer="PylonTech" Capacity="9000"  Yield="94" MaxCurrent="120" MaxVolt="48">										 | ||||||
| 		</Battery>	 | 		</Battery>	 | ||||||
| 		 |  | ||||||
| 	</Components>	 | 	</Components>	 | ||||||
| </Project> | </Project> | ||||||
| 		 | 		 | ||||||
| @@ -2,11 +2,7 @@ | |||||||
| 		 | 		 | ||||||
| <Project ProjectID="HA03" FriendlyName="@ProjectName" ProjectName="Tauberbischoffsheim SÜD" Established="2006" WattPeak="9840" ContentType="runnning"> | <Project ProjectID="HA03" FriendlyName="@ProjectName" ProjectName="Tauberbischoffsheim SÜD" Established="2006" WattPeak="9840" ContentType="runnning"> | ||||||
| 	<Components>	 | 	<Components>	 | ||||||
| 		<Panel PanelID="AIKO 01" FriendlyName="@PanelName" PanelName="AIKO 01 Solar T62B" Manufacturer="AIKO Solar" WattPeak="620" Height="2,70" Width="1,10" Weight="12" MaxVolt="67" MaxAmpere="11">			 | 		<Panel PanelID="AIKO 01" FriendlyName="@PanelName" PanelName="AIKO 01 Solar T62B" Manufacturer="AIKO Solar" WattPeak="620" Height="2,70" Width="1,10" Weight="12" MaxVolt="67" MaxAmpere="11"/>			 | ||||||
| 			<AdditionalData DataItem="Image"       DataValue="image.png"/> |  | ||||||
| 			<AdditionalData DataItem="Manual"      DataValue="manual.docx"/> |  | ||||||
| 			<AdditionalData DataItem="Certificate" DataValue="certificate.pdf"/> |  | ||||||
| 		</Panel> |  | ||||||
| 		<Panel PanelID="AIKO 02" FriendlyName="@PanelName" PanelName="AIKO 02 Solar X58C" Manufacturer="AIKO Solar" WattPeak="440" Height="1,70" Width="1,10" Weight="12" MaxVolt="42" MaxAmpere="11"/>			 | 		<Panel PanelID="AIKO 02" FriendlyName="@PanelName" PanelName="AIKO 02 Solar X58C" Manufacturer="AIKO Solar" WattPeak="440" Height="1,70" Width="1,10" Weight="12" MaxVolt="42" MaxAmpere="11"/>			 | ||||||
| 		<Panel PanelID="AIKO 03" FriendlyName="@PanelName" PanelName="AIKO 03 Solar T62B" Manufacturer="AIKO Solar" WattPeak="620" Height="2,70" Width="1,10" Weight="12" MaxVolt="67" MaxAmpere="11"/>			 | 		<Panel PanelID="AIKO 03" FriendlyName="@PanelName" PanelName="AIKO 03 Solar T62B" Manufacturer="AIKO Solar" WattPeak="620" Height="2,70" Width="1,10" Weight="12" MaxVolt="67" MaxAmpere="11"/>			 | ||||||
| 		<Panel PanelID="AIKO 04" FriendlyName="@PanelName" PanelName="AIKO 04 Solar X58C" Manufacturer="AIKO Solar" WattPeak="440" Height="1,70" Width="1,10" Weight="12" MaxVolt="42" MaxAmpere="11"/>			 | 		<Panel PanelID="AIKO 04" FriendlyName="@PanelName" PanelName="AIKO 04 Solar X58C" Manufacturer="AIKO Solar" WattPeak="440" Height="1,70" Width="1,10" Weight="12" MaxVolt="42" MaxAmpere="11"/>			 | ||||||
|   | |||||||
| @@ -6,8 +6,6 @@ | |||||||
|  |  | ||||||
| --> | --> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| <ItemTypes> | <ItemTypes> | ||||||
| 	<TreeParentType   RenderStyle="PlainStyle" 		 EditorType="LineEditType" ItemFlags="IsEnabled|IsDropEnabled" Icon="DirIcon" />	 | 	<TreeParentType   RenderStyle="PlainStyle" 		 EditorType="LineEditType" ItemFlags="IsEnabled|IsDropEnabled" Icon="DirIcon" />	 | ||||||
| 	<TreeChildType    RenderStyle="PlainStyle"       EditorType="LineEditType" ItemFlags="IsEnabled" Icon="MediaPlay"/> | 	<TreeChildType    RenderStyle="PlainStyle"       EditorType="LineEditType" ItemFlags="IsEnabled" Icon="MediaPlay"/> | ||||||
| @@ -15,6 +13,7 @@ | |||||||
| 	<HeaderType 	  RenderStyle="HeaderStyle" 	 EditorType="LineEditType" ItemFlags="IsEnabled"/> | 	<HeaderType 	  RenderStyle="HeaderStyle" 	 EditorType="LineEditType" ItemFlags="IsEnabled"/> | ||||||
| 	<HiddenType 	  RenderStyle="HiddenStyle"/> | 	<HiddenType 	  RenderStyle="HiddenStyle"/> | ||||||
| 	<StaticType 	  RenderStyle="PlainStyle"/> | 	<StaticType 	  RenderStyle="PlainStyle"/> | ||||||
|  | 	<FilePickerType	  RenderStyle="PlainStyle"    	 EditorType="LineEditType" ItemFlags="IsEnabled|IsEditable|IsSelectable" Icon="DirIcon" /> | ||||||
| 	<PlainType 		  RenderStyle="PlainStyle"    	 EditorType="LineEditType" ItemFlags="IsEnabled|IsEditable|IsSelectable"/> | 	<PlainType 		  RenderStyle="PlainStyle"    	 EditorType="LineEditType" ItemFlags="IsEnabled|IsEditable|IsSelectable"/> | ||||||
| 	<ValueType 		  RenderStyle="FormattedStyle"   EditorType="LineEditType" ItemFlags="IsEnabled|IsEditable|IsSelectable" UnitType="Coulomb"/> | 	<ValueType 		  RenderStyle="FormattedStyle"   EditorType="LineEditType" ItemFlags="IsEnabled|IsEditable|IsSelectable" UnitType="Coulomb"/> | ||||||
| 	<CheckableType 	  RenderStyle="FormattedStyle"   EditorType="LineEditType" ItemFlags="IsEnabled|IsEditable|IsSelectable" />	 | 	<CheckableType 	  RenderStyle="FormattedStyle"   EditorType="LineEditType" ItemFlags="IsEnabled|IsEditable|IsSelectable" />	 | ||||||
| @@ -56,8 +55,6 @@ | |||||||
| 	</Section> | 	</Section> | ||||||
| </DocumentTreeModel> | </DocumentTreeModel> | ||||||
| 	 | 	 | ||||||
|  |  | ||||||
|  |  | ||||||
| <DocumentDetailsModel> | <DocumentDetailsModel> | ||||||
| 	 | 	 | ||||||
| 	<Section ContentType="Panel" CatchBlock="TagName=Panel"> | 	<Section ContentType="Panel" CatchBlock="TagName=Panel"> | ||||||
| @@ -119,23 +116,28 @@ | |||||||
| 			<MaxVolt Caption="max. Volt" ItemType="HeaderType" />		 | 			<MaxVolt Caption="max. Volt" ItemType="HeaderType" />		 | ||||||
| 		</Header> | 		</Header> | ||||||
| 		<ModelSheet> | 		<ModelSheet> | ||||||
| 			<BatteryID ItemType="PlainType" /> | 			<BatteryID ItemType="PlainType"> | ||||||
|  | 				<Images> | ||||||
|  | 					<FrontView ItemType="FilePickerType"/> | ||||||
|  | 					<PackageView ItemType="FilePickerType"/> | ||||||
|  | 					<InstalledView ItemType="FilePickerType"/> | ||||||
|  | 				</Images > | ||||||
|  | 				<Documents> | ||||||
|  | 					<Manual ItemType="FilePickerType"/> | ||||||
|  | 					<Installation ItemType="FilePickerType"/>					 | ||||||
|  | 				</Documents> | ||||||
|  | 				<Certificates> | ||||||
|  | 					<Certificate ItemType="FilePickerType"/> | ||||||
|  | 					<Conformity ItemType="FilePickerType"/>					 | ||||||
|  | 				</Certificates>	 | ||||||
|  | 			</BatteryID> | ||||||
| 			<BatteryName ItemType="PlainType" /> | 			<BatteryName ItemType="PlainType" /> | ||||||
| 			<Manufacturer ItemType="PlainType" /> | 			<Manufacturer ItemType="PlainType" /> | ||||||
| 			<Capacity ItemType="ValueType"  UnitType="Wh"/> | 			<Capacity ItemType="ValueType"  UnitType="Wh"/> | ||||||
| 			<Yield ItemType="PercentageType" UnitType="%"/> | 			<Yield ItemType="PercentageType" UnitType="%"/> | ||||||
| 			<MaxCurrent ItemType="ValueType"  UnitType="A"> | 			<MaxCurrent ItemType="ValueType"  UnitType="A"/>			 | ||||||
| 				<SubType ItemType="PlainType"/>			 |  | ||||||
| 			</MaxCurrent> |  | ||||||
| 			<MaxVolt ItemType="ValueType"  UnitType="V"/>	 | 			<MaxVolt ItemType="ValueType"  UnitType="V"/>	 | ||||||
| 				 | 				 | ||||||
| 			<firz ItemType="PlainType"/> |  | ||||||
|  |  | ||||||
| 			<Image DataValue="image.png" ItemType="PlainType"/> |  | ||||||
| 			<Manual DataValue="manual.docx" ItemType="PlainType"/> |  | ||||||
| 			<Certificate DataValue="certificate.pdf" ItemType="PlainType"/> |  | ||||||
| 			 |  | ||||||
| 			 |  | ||||||
| 		</ModelSheet> | 		</ModelSheet> | ||||||
| 	</Section> | 	</Section> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,22 +1,24 @@ | |||||||
| <?xml version="1.0" encoding="UTF-8"?> | <?xml version="1.0" encoding="UTF-8"?> | ||||||
| <Components> | <Components> | ||||||
|     <Panel FriendlyName="@PanelName" Height="2,70" Manufacturer="Jingli Solar" MaxAmpere="11" MaxVolt="67" PanelID="Jingli 01" PanelName="Jingli 01 Solar T62B" WattPeak="620" Weight="12" Width="1,10"> |     <Panel FriendlyName="@PanelName" Height="2,70" Manufacturer="JA Solar 1 XX" MaxAmpere="11" MaxVolt="67" PanelID="#1 JA 01" PanelName="JA 01 Solar T62B" WattPeak="620" Weight="12" Width="1,10"/> | ||||||
|         <AdditionalData DataItem="Image" DataValue="image.png"/> |     <Panel FriendlyName="@PanelName" Height="1,70" Manufacturer="JA Solar 2" MaxAmpere="11" MaxVolt="42" PanelID="#2 JA 02" PanelName="JA 02 Solar X58C" WattPeak="440" Weight="12" Width="1,10"/> | ||||||
|         <AdditionalData DataItem="Manual" DataValue="manual.docx"/> |     <Panel FriendlyName="@PanelName" Height="2,70" Manufacturer="JA Solar 3" MaxAmpere="11" MaxVolt="67" PanelID="#3 JA 03" PanelName="JA 03 Solar T62B" WattPeak="620" Weight="12" Width="1,10"/> | ||||||
|         <AdditionalData DataItem="Certificate" DataValue="certificate.pdf"/> |     <Panel FriendlyName="@PanelName" Height="1,70" Manufacturer="JA Solar 4" MaxAmpere="11" MaxVolt="42" PanelID="#4 JA 04" PanelName="JA 04 Solar X58C" WattPeak="440" Weight="12" Width="1,10"/> | ||||||
|     </Panel> |     <Panel FriendlyName="@PanelName" Height="1,70" Manufacturer="JA Solar 5" MaxAmpere="11" MaxVolt="42" PanelID="#5 JA 05" PanelName="moo" WattPeak="440" Weight="12" Width="1,10"/> | ||||||
|     <Panel FriendlyName="@PanelName" Height="1,70" Manufacturer="Jingli Solar" MaxAmpere="11" MaxVolt="42" PanelID="Jingli 02" PanelName="Jingli 02 Solar X58C" WattPeak="440" Weight="12" Width="1,10"/> |     <Panel FriendlyName="@PanelName" Height="1,70" Manufacturer="JA Solar 6" MaxAmpere="11" MaxVolt="42" PanelID="#6 JA 06" PanelName="JA 06 Solar X58C" WattPeak="440" Weight="12" Width="1,10"/> | ||||||
|     <Panel FriendlyName="@PanelName" Height="2,70" Manufacturer="Jingli Solar" MaxAmpere="11" MaxVolt="67" PanelID="Jingli 03" PanelName="Jingli 03 Solar T62B" WattPeak="620" Weight="12" Width="1,10"/> |     <Inverter FriendlyName="@InverterName" InverterID="#1 HM600 01" InverterName="01 HM600 S2 TMax" Manufacturer="HoyMiles" MaxPowerInput="3000,00" MaxPowerInputChoice="2000;4000;6000" MaxPowerOutput="600" NumStrings="2" Weight="28"/> | ||||||
|     <Inverter FriendlyName="@InverterName" InverterID="HM600 01" InverterName="01 HM600 S2 TMax" Manufacturer="HoyMiles" MaxPowerInput="4000" MaxPowerInputChoice="2000;4000;6000" MaxPowerOutput="600" NumStrings="2" Weight="28"/> |     <Inverter FriendlyName="@InverterName" InverterID="#2 HM800 02" InverterName="02 HM800 S2 TMax" Manufacturer="HoyMiles" MaxPowerInput="4000" MaxPowerInputChoice="4000;6000;8000" MaxPowerOutput="800" NumStrings="2" Weight="29"/> | ||||||
|     <Inverter FriendlyName="@InverterName" InverterID="HM800 02" InverterName="02 HM800 S2 TMax" Manufacturer="HoyMiles" MaxPowerInput="4000" MaxPowerInputChoice="2000;4000;6000" MaxPowerOutput="800" NumStrings="2" Weight="29"/> |     <Inverter FriendlyName="@InverterName" InverterID="#3 HM1600 03" InverterName="03 HM1600 S4 TMax" Manufacturer="HoyMiles" MaxPowerInput="9000,00" MaxPowerInputChoice="6000;8000;10000" MaxPowerOutput="1600" NumStrings="4" Weight="32"/> | ||||||
|     <Inverter FriendlyName="@InverterName" InverterID="HM1600 03" InverterName="03 HM1600 S4 TMax" Manufacturer="HoyMiles" MaxPowerInput="4000" MaxPowerInputChoice="2000;4000;6000" MaxPowerOutput="1600" NumStrings="4" Weight="32"/> |     <Inverter FriendlyName="@InverterName" InverterID="#4 D12K 04" InverterName="04 HM600 S2 TMax" Manufacturer="Deye" MaxPowerInput="8000" MaxPowerInputChoice="6000;8000;12000" MaxPowerOutput="600" NumStrings="2" Weight="28"/> | ||||||
|     <Inverter FriendlyName="@InverterName" InverterID="D12K 04" InverterName="04 HM600 S2 TMax" Manufacturer="Deye" MaxPowerInput="4000" MaxPowerInputChoice="2000;4000;6000" MaxPowerOutput="600" NumStrings="2" Weight="28"/> |     <Battery BatteryID="#1 BYD 01" BatteryName="01 BYD T01 Stackable" Capacity="4500" FriendlyName="@BatteryName" Manufacturer="BYD" MaxCurrent="120" MaxVolt="48" Yield="90"> | ||||||
|     <Battery BatteryID="BYD 01" BatteryName="FIRZ!" Capacity="4500" FriendlyName="@BatteryName" Manufacturer="BYD" MaxCurrent="120" MaxVolt="48" Yield="88"> |         <Images FrontView="image.png" InstalledView="installed.png" PackageView="package.png"/> | ||||||
|         <AdditionalData DataItem="Image" DataValue="image.png"/> |         <Documents Installation="installation.pdf" Manual="manual.docx"/> | ||||||
|         <AdditionalData DataItem="Manual" DataValue="manual.docx"/> |         <Certificates Certificate="certificate.pdf" Conformity="conformity.pdf"/> | ||||||
|         <AdditionalData DataItem="Certificate" DataValue="certificate.pdf"/> |  | ||||||
|     </Battery> |     </Battery> | ||||||
|     <Battery BatteryID="BYD 04" BatteryName="04 BYD T02 Stackable" Capacity="9000" FriendlyName="@BatteryName" Manufacturer="BYD" MaxCurrent="120" MaxVolt="48" Yield="32"/> |     <Battery BatteryID="#2 BYD 02" BatteryName="02 BYD T02 Stackable" Capacity="9000" FriendlyName="@BatteryName" Manufacturer="BYD" MaxCurrent="120" MaxVolt="48" Yield="94"/> | ||||||
|     <Battery BatteryID="GroWatt 05 G2K" BatteryName="05 BYD T01 Stackable" Capacity="4500" FriendlyName="@BatteryName" Manufacturer="PylonTech" MaxCurrent="120" MaxVolt="48" Yield="46"/> |     <Battery BatteryID="#3 BYD 03" BatteryName="03 BYD T01 Stackable" Capacity="4500" FriendlyName="@BatteryName" Manufacturer="BYD" MaxCurrent="120" MaxVolt="48" Yield="86"/> | ||||||
|     <Battery BatteryID="GroWatt 06 G4K" BatteryName="06 BYD T02 Stackable" Capacity="9000" FriendlyName="@BatteryName" Manufacturer="PylonTech" MaxCurrent="120" MaxVolt="48" Yield="94"/> |     <Battery BatteryID="#4 BYD 04" BatteryName="04 BYD T02 Stackable" Capacity="9000" FriendlyName="@BatteryName" Manufacturer="BYD" MaxCurrent="120" MaxVolt="48" Yield="91"/> | ||||||
|  |     <Battery BatteryID="#5 GroWatt 05 G2K" BatteryName="05 BYD T01 Stackable" Capacity="4500" FriendlyName="@BatteryName" Manufacturer="GroWatt" MaxCurrent="120" MaxVolt="48" Yield="94"/> | ||||||
|  |     <Battery BatteryID="#6 GroWatt 06 G4K" BatteryName="06 BYD T02 Stackable" Capacity="9000" FriendlyName="@BatteryName" Manufacturer="GroWatt" MaxCurrent="120" MaxVolt="48" Yield="49"/> | ||||||
|  |     <Battery BatteryID="#7 Pyne 07 G4K" BatteryName="07 Pyne K7 Stackable" Capacity="9000" FriendlyName="@BatteryName" Manufacturer="PyNe" MaxCurrent="120" MaxVolt="48" Yield="68"/> | ||||||
| </Components> | </Components> | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user