From 5fc9871b5f2fb608553822dac32a7ae1bc8c3f1d Mon Sep 17 00:00:00 2001 From: Mr-Auto <36127424+Mr-Auto@users.noreply.github.com> Date: Sat, 4 May 2024 22:06:10 +0200 Subject: [PATCH] fix wstring, fix crash when clicking empty field in ThreadsView, improve the look of memory tree, fix other small stuff in memory tree, rename `ThemeInfoName` -> `ThemeInfoPointer` --- include/Configuration.h | 2 +- include/Data/StdString.h | 2 +- include/QtHelpers/TreeViewMemoryFields.h | 11 +- resources/Spelunky2.json | 2 +- resources/spelunky2.qrc | 21 +- resources/stylesheet-branch-closed.png | Bin 0 -> 4873 bytes resources/stylesheet-branch-end.png | Bin 0 -> 182 bytes resources/stylesheet-branch-more.png | Bin 0 -> 136 bytes resources/stylesheet-branch-open.png | Bin 0 -> 687 bytes resources/stylesheet-vline.png | Bin 0 -> 124 bytes src/Configuration.cpp | 2 +- src/QtHelpers/StyledItemDelegateHTML.cpp | 19 +- src/QtHelpers/TreeViewMemoryFields.cpp | 396 ++++++++++------------- src/Views/ViewThreads.cpp | 11 +- 14 files changed, 226 insertions(+), 240 deletions(-) create mode 100644 resources/stylesheet-branch-closed.png create mode 100644 resources/stylesheet-branch-end.png create mode 100644 resources/stylesheet-branch-more.png create mode 100644 resources/stylesheet-branch-open.png create mode 100644 resources/stylesheet-vline.png diff --git a/include/Configuration.h b/include/Configuration.h index 8e6ad759..f967bfdf 100644 --- a/include/Configuration.h +++ b/include/Configuration.h @@ -115,7 +115,7 @@ namespace S2Plugin EntitySubclass, // a subclass of an entity defined in json DefaultStructType, // a struct defined in json UndeterminedThemeInfoPointer, // used to look up the theme pointer in the levelgen and show the correct theme name - ThemeInfoName, // same as above, but does not add struct tree + ThemeInfoPointer, // same as above, but does not add struct tree LevelGenRoomsPointer, // used to make the level gen rooms title clickable LevelGenRoomsMetaPointer, // used to make the level gen rooms title clickable JournalPagePointer, // used to make journal page in vector clickable diff --git a/include/Data/StdString.h b/include/Data/StdString.h index 5e8d2e5d..4a57d94d 100644 --- a/include/Data/StdString.h +++ b/include/Data/StdString.h @@ -76,5 +76,5 @@ namespace S2Plugin }; using StdString = StdBasicString; - using StdWstring = StdBasicString; + using StdWstring = StdBasicString; } // namespace S2Plugin diff --git a/include/QtHelpers/TreeViewMemoryFields.h b/include/QtHelpers/TreeViewMemoryFields.h index c64aadfc..adfc021f 100644 --- a/include/QtHelpers/TreeViewMemoryFields.h +++ b/include/QtHelpers/TreeViewMemoryFields.h @@ -46,11 +46,16 @@ namespace S2Plugin public: TreeViewMemoryFields(QWidget* parent = nullptr); - void addMemoryFields(const std::vector& fields, const std::string& mainName, uintptr_t structAddr, size_t initialDelta = 0, QStandardItem* parent = nullptr); - QStandardItem* addMemoryField(const MemoryField& field, const std::string& fieldNameOverride, uintptr_t memoryAddress, size_t delta, QStandardItem* parent = nullptr); + void addMemoryFields(const std::vector& fields, const std::string& mainName, uintptr_t structAddr, size_t initialDelta = 0, uint8_t deltaPrefixCount = 0, + QStandardItem* parent = nullptr); + QStandardItem* addMemoryField(const MemoryField& field, const std::string& fieldNameOverride, uintptr_t memoryAddress, size_t delta, uint8_t deltaPrefixCount = 0, + QStandardItem* parent = nullptr); void clear(); void updateTableHeader(bool restoreColumnWidths = true); - void setEnableChangeHighlighting(bool b) noexcept; + void setEnableChangeHighlighting(bool b) noexcept + { + mEnableChangeHighlighting = b; + } void updateTree(uintptr_t newAddr, uintptr_t newComparisonAddr = 0, bool initial = false); void updateRow(int row, std::optional newAddr = std::nullopt, std::optional newAddrComparison = std::nullopt, QStandardItem* parent = nullptr, diff --git a/resources/Spelunky2.json b/resources/Spelunky2.json index f8712b58..0550e2cd 100644 --- a/resources/Spelunky2.json +++ b/resources/Spelunky2.json @@ -9051,7 +9051,7 @@ { "field": "padding3?", "type": "UnsignedDword" }, { "field": "sub_theme", - "type": "ThemeInfoName", + "type": "ThemeInfoPointer", "comment": "for cosmic ocean" }, { "field": "unknown3", "type": "UnsignedDword" }, diff --git a/resources/spelunky2.qrc b/resources/spelunky2.qrc index c21da040..37c9d783 100644 --- a/resources/spelunky2.qrc +++ b/resources/spelunky2.qrc @@ -1,5 +1,20 @@ - - caveman.png - + + caveman.png + + + stylesheet-vline.png + + + stylesheet-branch-more.png + + + stylesheet-branch-end.png + + + stylesheet-branch-closed.png + + + stylesheet-branch-open.png + \ No newline at end of file diff --git a/resources/stylesheet-branch-closed.png b/resources/stylesheet-branch-closed.png new file mode 100644 index 0000000000000000000000000000000000000000..3203da14dd2e912b5c033ffa04a77df449f8aca0 GIT binary patch literal 4873 zcmeHKdr%YC8ebj~1QZ_?pDjz2h+>k>lVnXmBq*RkULx9yve|54Aup1JL>wR>TG9H# zcSU6c@mZ!QRz+G+QLCV$ZT0$~*9ui@QB)ALw0chhDoi_b$C=yzNM^IU-}%1Z`M%%z zoo{mTBf@76av9?SLC~Pk5Lpzs!`92$3H%x}GTOndHX|mEio#5=-k?*FY67Mf>j{`J zlPU-@-_k?`4z7irtGZTZ=5qJF9Cz|DqibpA>)j2d8Z>`AqH;Xl(Uo;N#WIX`d2X1m z_vGWL#br6Npk0OZIXmGk{OTXhc&*!9WPcy)Q+KDUojpeFsa z8}OX#rR_uc_b)g;52w85EV(uGb))P2VON5;xi$eC&5 z&Mt~xwR;3@>I!GozTAY7BT}c#oszrkT5Qsfr<$*^ldi2RIWBT38$b*2I5?Pfxe`6@ zl6#l7;>gksyH{;|SzXdVKex0gtLkLu@g+3p)h^MO%6y8FH?LoLzE-WB+maOwWwm4W z#lv_nWEmWXE6>v0Jr}joC3RhQb`>wWu-)xxer;swSdV8;&ss|N7=M~LxV)fb$gbgY z#!!LxW|q~ls#96$l&AIU)`wmi{@IGN3R>OA^_&7@4!o`O7hr7*Q3TZBN~&zMgPGY1nk?yK>9qn2p1i_~NvwX4r8Sg1?+d3-{` z*yG2wDbGLApKx%K{#xKAF4$EX;RCfMw#&r{Pdy&S8<(U!ZfdvtE@EDtU(G2mWMN55 z+FDa5)zQ(qEjBx#sL61GtP>Xl-KCR2Uvcu;D6Z46Fr`jGu*@1g5FP|c0?c|0Pa!B+ zK_rq|DZTY%H613EQu;iB9FgmTh$J#3!$3r5gva0+DY#fk5Ab)9m{92dN1I7nw8U_Oh_LYTp3GM!8JcY!4ar3#Ib&FFyuZ&G>^ zMd?vC+hj7aOgxs(kjUnU#bP$XWplYq;K4L5)>4?6sWpzbLUd!u2qSJF^%SYo!d6U7 zp-ZErbUGM^-{z;$%jNIjwZ-1x_$A1tMS)1c%^bIF~CzguS3bwMGil;)E3nfU`(|Bj$;5g;*$JBA6A2 z!x1tSe6f&;BZQyAPbJ`qgrZ&$a||R6 ztV#s{wg^xS8e||aN@s}C>C{rXH7VE{`EFPaHk1;hFd0S>02JZ!QAB_uyciA->epQpU%>=`?ZGmEVcL_!; zolx5H1X$fecoL>fBtZY@5$sz#`8Q(Vs5pq&Pb6R}1spMx50t@F2nBqgCnbW2aFIa4 z=l4Z7>Qs~oGZ2A^fJeX;5U7nS*r#WyCiV3w_5hAm@H{{om+vd4HH9wBR3*2^^K~_z#A`G0RCYYo-i(WBnF=vv&iq4Ah6rH$sr> zaO-6UeNi|H7->|fJec-yfP<^Aha$1U8G`I9LS=z5ac3THxio7oHsoA(nCFZfm14~N z3eMq>1B47eRx*q}Ddl@%(ulx=58ac#r#T+orx~(OHZ-sK#vcPxnSulIC~U@ErjLBU zZ3&IFcgdB-?!iMv`q7t9w0+xFU&U+lr)_TboE7YW^FDR2uxHjRD2mONJ1nw9!mUrD zvp=$I(>S3yUzQ)e`uoMLSp(ZEyKeu-K0#IYi@92zXptcHvnM56l8Yj5-gG#0y(_Tj zcH>B{19D_;&HP{~&CTx5C;r>*D_uk`6%M|Z)VrHcUU77YXnxRK1>H&O%>Vqw;h`e; zAMA9Nlqdcpj~$$HZMerc~iS8uR`85aLn=nm(a`bV3}bknl84N9Eb2*rEB zmWj8HoUI>I!2K#ZqqZ@7f=uAn4z03J6?wbzYIxEo;r0zvq#o;Tc@@oDyF-{`vR zGqW0#RldUukBtzwhUQBu=YNVne(4LD-Z)lW%lv7lzVZIbLU*Ua*v~q~rFE_)(0orC)-qcdUcPfc&G0#-T2Yt<7pB(0k2XUTw^BAa@ zyCldjn4v~I(#klsLO;e{w&G|gP}tJb#WBR=cyhu4mcAxNo+FKcQx|KwhxVyV?Ve)L zz*IK1wMoLE!Dwo3=Hy<(2FAl060D08M0g?@7(yn?%xc=$H6LgQgQu&X%Q~loCIF7Q BI8p!r literal 0 HcmV?d00001 diff --git a/resources/stylesheet-branch-more.png b/resources/stylesheet-branch-more.png new file mode 100644 index 0000000000000000000000000000000000000000..664ad44740ef5729cfa2ca5ca8395ebdf8cb2277 GIT binary patch literal 136 zcmeAS@N?(olHy`uVBq!ia0vp^5;M1%lf!(`5Nmw&E1&}Q zk|4ie1}`^bv!LK9&GY?$Leictjv*Y;$tetsZa>)*y2R$Br1%LGR;@U~$WV7luyf{a R_C-Kt44$rjF6*2UngE&%B+>u? literal 0 HcmV?d00001 diff --git a/resources/stylesheet-branch-open.png b/resources/stylesheet-branch-open.png new file mode 100644 index 0000000000000000000000000000000000000000..60470061c8f2efc712caedbd2d776de8cab544ae GIT binary patch literal 687 zcmV;g0#N;lP)Px#1ZP1_K>z@;j|==^1poj532;bRa{vGf6951U69E94oEQKA0z^qfK~z{r?U%WV z6hRn(M^O{kD^x)YyuiTw5j52G9ZUrSQFD(}6hp5!P%-iZUqX=$R(C~A6ucvk?<@OX zi(#s_dU`;F{otdgYW}LOuKuU0X0XJ+W?6HsWhWlP!?*=EqP)Y`_yq6ZOPnYN@G<_X z)PWqt-mJi@_!I5;J>EcPVF#{h+}6}xM`Mh&%{5$!W3e9NslVX`T#Jig%z6BZF&i7O z6Fzy|d>;2>SN7o?+Q(DuG!s|QhHr4Q6=Sjq?Zdu&9aZecALz6nvZ5y@qp!o9s7lQH z=!h?-(qu3v?9+T`Wjh-90oPeEAxzBpb-0!+u)Mhj)6MlBe*E)sfvM(N!*J{;bb*q% z$4DNcqZ|&}inDksmD(}JSli?rMf>*GRpbR4v@4bVassW_wn6w^#Mvgtskh#H6#0w> zjhgPf^HGDaE>_G@->*i$`;tlp+6Ew{L6quqss`{J4H|w0IcG7ynZ}c!!xB}m-~k$R zJeA7iPaKw>JaDrz?olcgIE@C~N~L!4nP_|Rz+KeXGpSU-ZPlRKh3I7CV&n^v+juKZ zk%5;|X)?H#+h^+<5V-Za8GXmSR`g^Ky0Mw4o~)|4jE>lIUf=Eb72204t7<%u#Io#x zWSsMzGilqHdi(5zN3zHVUxztSlbFW1l{@XWeTMcm7LMB2VNTeYzmD7JI^kj!hf)_A z>u#bk#@fb{)hg_b$ry`Njm})vIkM{?6AvtIt4Gj3CXt|T(F3SAzKb)s#Q(rx@EfhS Vv~Oi-85RHl002ovPDHLkV1k%UMi>A9 literal 0 HcmV?d00001 diff --git a/resources/stylesheet-vline.png b/resources/stylesheet-vline.png new file mode 100644 index 0000000000000000000000000000000000000000..8f0c336fd8c07b851766be334a0b3ab2744afaff GIT binary patch literal 124 zcmeAS@N?(olHy`uVBq!ia0vp^fk14<#0(_2UW&B=Qak}ZA+G=b|4$C{g`lg;Rdj*k zEG0pH!3;H4#wCm%2Z4NPPZ!4!3;*PV2F7y|Y&r=cB@k%Q#l#>uQ|xhh=&~H3G6qjq KKbLh*2~7Z6M #include +#include #include void S2Plugin::StyledItemDelegateHTML::paint(QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index) const @@ -14,7 +16,7 @@ void S2Plugin::StyledItemDelegateHTML::paint(QPainter* painter, const QStyleOpti options.text = ""; options.widget->style()->drawControl(QStyle::CE_ItemViewItem, &options, painter); - + QSize iconSize = options.icon.actualSize(options.rect.size()); if (mCenterVertically) { doc.setTextWidth(options.rect.width()); @@ -23,10 +25,19 @@ void S2Plugin::StyledItemDelegateHTML::paint(QPainter* painter, const QStyleOpti } else { - painter->translate(options.rect.left(), options.rect.top() - 2); + painter->translate(options.rect.left() + iconSize.width(), options.rect.top() - 2); } - QRect clip(0, 0, options.rect.width(), options.rect.height()); - doc.drawContents(painter, clip); + QRect clip(0, 0, options.rect.width() + iconSize.width(), options.rect.height()); + // doc.drawContents(painter, clip); + + painter->setClipRect(clip); + QAbstractTextDocumentLayout::PaintContext ctx; + auto newColor = index.data(Qt::TextColorRole); + if (!newColor.isNull()) + ctx.palette.setColor(QPalette::Text, newColor.value()); + + ctx.clip = clip; + doc.documentLayout()->draw(painter, ctx); painter->restore(); } diff --git a/src/QtHelpers/TreeViewMemoryFields.cpp b/src/QtHelpers/TreeViewMemoryFields.cpp index 8707bc4a..f39f158b 100644 --- a/src/QtHelpers/TreeViewMemoryFields.cpp +++ b/src/QtHelpers/TreeViewMemoryFields.cpp @@ -47,17 +47,38 @@ S2Plugin::TreeViewMemoryFields::TreeViewMemoryFields(QWidget* parent) : QTreeVie setDragEnabled(true); setAcceptDrops(false); + setStyleSheet("QTreeView::branch:has-siblings:!adjoins-item {\ + border-image: url(:/images/vline.png) 0;\ +}\ +QTreeView::branch:has-siblings:adjoins-item {\ + border-image: url(:/images/branch-more.png) 0;\ +}\ +QTreeView::branch:!has-children:!has-siblings:adjoins-item {\ + border-image: url(:/images/branch-end.png) 0;\ +}\ +QTreeView::branch:has-children:!has-siblings:closed,\ +QTreeView::branch:closed:has-children:has-siblings {\ + border-image: none;\ + image: url(:/images/branch-closed.png);\ +}\ +QTreeView::branch:open:has-children:!has-siblings,\ +QTreeView::branch:open:has-children:has-siblings {\ + border-image: none;\ + image: url(:/images/branch-open.png);\ +}"); + QObject::connect(this, &QTreeView::clicked, this, &TreeViewMemoryFields::cellClicked); } -void S2Plugin::TreeViewMemoryFields::addMemoryFields(const std::vector& fields, const std::string& mainName, uintptr_t structAddr, size_t initialDelta, QStandardItem* parent) +void S2Plugin::TreeViewMemoryFields::addMemoryFields(const std::vector& fields, const std::string& mainName, uintptr_t structAddr, size_t initialDelta, uint8_t deltaPrefixCount, + QStandardItem* parent) { size_t currentOffset = structAddr; size_t currentDelta = initialDelta; for (auto& field : fields) { - addMemoryField(field, mainName + "." + field.name, currentOffset, currentDelta, parent); + addMemoryField(field, mainName + "." + field.name, currentOffset, currentDelta, deltaPrefixCount, parent); auto size = field.get_size(); currentDelta += size; if (structAddr != 0) @@ -65,9 +86,11 @@ void S2Plugin::TreeViewMemoryFields::addMemoryFields(const std::vector QStandardItem* + auto createAndInsertItem = [&delta, &deltaPrefixCount](const MemoryField& field, const std::string& fieldNameUID, QStandardItem* itemParent, uintptr_t memAddr, + bool showDelta = true) -> QStandardItem* { auto itemFieldName = new QStandardItem(); itemFieldName->setEditable(false); @@ -81,42 +104,40 @@ QStandardItem* S2Plugin::TreeViewMemoryFields::addMemoryField(const MemoryField& itemFieldValue->setEditable(false); if (field.isPointer == false) // if it's pointer, we set it on first update itemFieldValue->setData(memAddr, gsRoleMemoryAddress); - itemFieldValue->setData("", Qt::DisplayRole); auto itemFieldValueHex = new QStandardItem(); itemFieldValueHex->setEditable(false); - itemFieldValueHex->setData("", Qt::DisplayRole); auto itemFieldComparisonValue = new QStandardItem(); itemFieldComparisonValue->setEditable(false); - itemFieldComparisonValue->setData("", Qt::DisplayRole); auto itemFieldComparisonValueHex = new QStandardItem(); itemFieldComparisonValueHex->setEditable(false); - itemFieldComparisonValueHex->setData("", Qt::DisplayRole); auto itemFieldMemoryOffset = new QStandardItem(); itemFieldMemoryOffset->setEditable(false); - if (memAddr == 0) - itemFieldMemoryOffset->setData("", Qt::DisplayRole); - else + if (memAddr != 0) { itemFieldMemoryOffset->setData(QString::asprintf("0x%016llX", memAddr), Qt::DisplayRole); - // for click event. I could just use the itemFieldName(gsRoleMemoryAddress) for it, but doing it this way for potential itemComparisonFieldMemoryOffset in future + // for click event. we could just use the itemFieldName(gsRoleMemoryAddress), but doing it this way for potential itemComparisonFieldMemoryOffset in the future itemFieldMemoryOffset->setData(memAddr, gsRoleRawValue); } auto itemFieldMemoryOffsetDelta = new QStandardItem(); itemFieldMemoryOffsetDelta->setEditable(false); - if (showDelta) + if (showDelta) // this should only ever be false for flag field { - itemFieldMemoryOffsetDelta->setData(QString::asprintf("+0x%llX", delta), Qt::DisplayRole); + QString text; + if (deltaPrefixCount > 0) + { + text = QString(deltaPrefixCount - 1, QChar(0x2502)); // '│' + text += QChar(0x2514); // '└' + text += QChar(0x2192); // '→' + } + text += QString::asprintf("+0x%llX", delta); + itemFieldMemoryOffsetDelta->setData(text, Qt::DisplayRole); itemFieldMemoryOffsetDelta->setData(delta, gsRoleRawValue); } - else - { - itemFieldMemoryOffsetDelta->setData("", Qt::DisplayRole); // this should only ever happen for flag field - } auto itemFieldComment = new QStandardItem(); itemFieldComment->setEditable(false); @@ -145,7 +166,7 @@ QStandardItem* S2Plugin::TreeViewMemoryFields::addMemoryField(const MemoryField& { parent = mModel->invisibleRootItem(); } - + uint8_t flags = 0; QStandardItem* returnField = nullptr; switch (field.type) { @@ -184,7 +205,7 @@ QStandardItem* S2Plugin::TreeViewMemoryFields::addMemoryField(const MemoryField& case MemoryFieldType::LevelGenRoomsPointer: case MemoryFieldType::LevelGenRoomsMetaPointer: case MemoryFieldType::JournalPagePointer: - case MemoryFieldType::ThemeInfoName: + case MemoryFieldType::ThemeInfoPointer: case MemoryFieldType::UTF16Char: case MemoryFieldType::IPv4Address: { @@ -213,10 +234,19 @@ QStandardItem* S2Plugin::TreeViewMemoryFields::addMemoryField(const MemoryField& break; } case MemoryFieldType::Flags32: + flags = 32; + [[fallthrough]]; + case MemoryFieldType::Flags16: + if (flags == 0) + flags = 16; + [[fallthrough]]; + case MemoryFieldType::Flags8: { + if (flags == 0) + flags = 8; + auto flagsParent = createAndInsertItem(field, fieldNameOverride, parent, memoryAddress); - flagsParent->setData(QVariant::fromValue(field.firstParameterType), gsRoleRefName); - for (uint8_t x = 1; x <= 32; ++x) + for (uint8_t x = 1; x <= flags; ++x) { MemoryField flagField; flagField.name = "flag_" + std::to_string(x); @@ -229,39 +259,10 @@ QStandardItem* S2Plugin::TreeViewMemoryFields::addMemoryField(const MemoryField& } auto flagFieldItem = createAndInsertItem(flagField, fieldNameOverride + "." + flagField.name, flagsParent, 0, showDelta); flagFieldItem->setData(x, gsRoleFlagIndex); - } - returnField = flagsParent; - break; - } - case MemoryFieldType::Flags16: - { - auto flagsParent = createAndInsertItem(field, fieldNameOverride, parent, memoryAddress, delta); - flagsParent->setData(QVariant::fromValue(field.firstParameterType), gsRoleRefName); - for (uint8_t x = 1; x <= 16; ++x) - { - MemoryField flagField; - flagField.name = "flag_" + std::to_string(x); - flagField.type = MemoryFieldType::Flag; - bool showDelta = x == 1 || x == 9; - delta += x == 9 ? 1 : 0; - auto flagFieldItem = createAndInsertItem(flagField, fieldNameOverride + "." + flagField.name, flagsParent, 0, showDelta); - flagFieldItem->setData(x, gsRoleFlagIndex); - } - returnField = flagsParent; - break; - } - case MemoryFieldType::Flags8: - { - auto flagsParent = createAndInsertItem(field, fieldNameOverride, parent, memoryAddress, delta); - flagsParent->setData(QVariant::fromValue(field.firstParameterType), gsRoleRefName); - for (uint8_t x = 1; x <= 8; ++x) - { - MemoryField flagField; - flagField.name = "flag_" + std::to_string(x); - flagField.type = MemoryFieldType::Flag; - bool showDelta = x == 1; // for 8 bit flag we only show the first - auto flagFieldItem = createAndInsertItem(flagField, fieldNameOverride + "." + flagField.name, flagsParent, 0, showDelta); - flagFieldItem->setData(x, gsRoleFlagIndex); + auto flagName = Configuration::get()->flagTitle(field.firstParameterType, x); + QString realFlagName = QString::fromStdString(flagName.empty() ? Configuration::get()->flagTitle("unknown", x) : flagName); // TODO: don't show unknown unless it was chosen in settings + + flagsParent->child(x - 1, gsColValue)->setData(realFlagName, Qt::DisplayRole); } returnField = flagsParent; break; @@ -269,7 +270,7 @@ QStandardItem* S2Plugin::TreeViewMemoryFields::addMemoryField(const MemoryField& case MemoryFieldType::UndeterminedThemeInfoPointer: { returnField = createAndInsertItem(field, fieldNameOverride, parent, memoryAddress); - addMemoryFields(Configuration::get()->typeFieldsOfDefaultStruct("ThemeInfoPointer"), fieldNameOverride, 0, 0, returnField); + addMemoryFields(Configuration::get()->typeFieldsOfDefaultStruct("ThemeInfoPointer"), fieldNameOverride, 0, 0, deltaPrefixCount + 1, returnField); break; } case MemoryFieldType::StdVector: @@ -277,9 +278,9 @@ QStandardItem* S2Plugin::TreeViewMemoryFields::addMemoryField(const MemoryField& returnField = createAndInsertItem(field, fieldNameOverride, parent, memoryAddress); returnField->setData(QVariant::fromValue(field.firstParameterType), gsRoleStdContainerFirstParameterType); if (field.isPointer) - addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, 0, 0, returnField); + addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, 0, 0, deltaPrefixCount + 1, returnField); else - addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, memoryAddress, delta, returnField); + addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, memoryAddress, delta, deltaPrefixCount, returnField); break; } @@ -289,25 +290,25 @@ QStandardItem* S2Plugin::TreeViewMemoryFields::addMemoryField(const MemoryField& returnField->setData(QVariant::fromValue(field.firstParameterType), gsRoleStdContainerFirstParameterType); returnField->setData(QVariant::fromValue(field.secondParameterType), gsRoleStdContainerSecondParameterType); if (field.isPointer) - addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, 0, 0, returnField); + addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, 0, 0, deltaPrefixCount + 1, returnField); else - addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, memoryAddress, delta, returnField); + addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, memoryAddress, delta, deltaPrefixCount, returnField); break; } case MemoryFieldType::EntitySubclass: { returnField = createAndInsertItem(field, fieldNameOverride, parent, 0); - addMemoryFields(Configuration::get()->typeFieldsOfEntitySubclass(field.jsonName), fieldNameOverride, memoryAddress, delta, returnField); + addMemoryFields(Configuration::get()->typeFieldsOfEntitySubclass(field.jsonName), fieldNameOverride, memoryAddress, delta, deltaPrefixCount, returnField); break; } case MemoryFieldType::DefaultStructType: { returnField = createAndInsertItem(field, fieldNameOverride, parent, memoryAddress); if (field.isPointer) - addMemoryFields(Configuration::get()->typeFieldsOfDefaultStruct(field.jsonName), fieldNameOverride, 0, 0, returnField); + addMemoryFields(Configuration::get()->typeFieldsOfDefaultStruct(field.jsonName), fieldNameOverride, 0, 0, deltaPrefixCount + 1, returnField); else - addMemoryFields(Configuration::get()->typeFieldsOfDefaultStruct(field.jsonName), fieldNameOverride, memoryAddress, delta, returnField); + addMemoryFields(Configuration::get()->typeFieldsOfDefaultStruct(field.jsonName), fieldNameOverride, memoryAddress, delta, deltaPrefixCount, returnField); break; } @@ -315,9 +316,9 @@ QStandardItem* S2Plugin::TreeViewMemoryFields::addMemoryField(const MemoryField& { returnField = createAndInsertItem(field, fieldNameOverride, parent, memoryAddress); if (field.isPointer) - addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, 0, 0, returnField); + addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, 0, 0, deltaPrefixCount + 1, returnField); else - addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, memoryAddress, delta, returnField); + addMemoryFields(Configuration::get()->typeFields(field.type), fieldNameOverride, memoryAddress, delta, deltaPrefixCount, returnField); break; } @@ -392,16 +393,16 @@ void S2Plugin::TreeViewMemoryFields::updateTree(uintptr_t newAddr, uintptr_t new // hope that the compiler can inline and optimise all of this 🙏 template inline std::optional updateField(QStandardItem* itemField, uintptr_t memoryAddress, QStandardItem* itemValue, const char* valueFormat, QStandardItem* itemValueHex, bool isPointer, - const char* hexFormat, bool updateBackground, bool resetBackgroundToTransparent, QColor& background) + const char* hexFormat, bool updateBackground, bool resetBackgroundToTransparent, const QColor& background) { std::optional value; if (memoryAddress == 0) { - itemValue->setData("", Qt::DisplayRole); + itemValue->setData({}, Qt::DisplayRole); if (!isPointer) - itemValueHex->setData("", Qt::DisplayRole); + itemValueHex->setData({}, Qt::DisplayRole); - itemValue->setData(QVariant{}, S2Plugin::gsRoleRawValue); + itemValue->setData({}, S2Plugin::gsRoleRawValue); itemField->setBackground(Qt::transparent); } else @@ -477,7 +478,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional QStandardItem* itemMemoryOffsetDelta = parent->child(row, gsColMemoryAddressDelta); auto deltaData = itemMemoryOffsetDelta->data(gsRoleRawValue); memoryOffset = newAddr.value() == 0 ? 0 : newAddr.value() + deltaData.toULongLong(); - if (!deltaData.isNull()) + if (!deltaData.isNull() && fieldType != MemoryFieldType::Flag) { itemMemoryOffset->setData(QString::asprintf("0x%016llX", memoryOffset), Qt::DisplayRole); itemMemoryOffset->setData(memoryOffset, gsRoleRawValue); @@ -503,15 +504,15 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional bool pointerUpdate = false; bool comparisonPointerUpdate = false; - bool comparisonPointerDifference = false; bool comparisonActive = activeColumns.test(gsColComparisonValue) || activeColumns.test(gsColComparisonValueHex); - uintptr_t newPointer = 0; - uintptr_t newComparisonPointer = 0; uintptr_t valueMemoryOffset = memoryOffset; // 0, memory offset or pointer value (no bad values) uintptr_t valueComparisonMemoryOffset = comparisonMemoryOffset; // 0, memory offset or pointer value (no bad values) if (isPointer) { + uintptr_t newPointer = 0; + uintptr_t newComparisonPointer = 0; + bool comparisonPointerDifference = false; // dealing with itemValueHex and itemComparisonValueHex for all pointers and check if the pointer changed auto checkAndUpdatePointer = [fieldType](uintptr_t& pointerValue, QStandardItem* valueHexField) -> bool { @@ -529,13 +530,14 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional pointerValue = 0; } else - newHexValue = QString::asprintf("0x%016llX", pointerValue); - - // TODO: if we could set the color separately, we could avoid this check, (we would also then not need the newPointer and newComparisonPointer to have scope on the whole function?) - if (fieldType != MemoryFieldType::CodePointer) // for Code pointer we paint it green instead { - valueHexField->setData(newHexValue, Qt::DisplayRole); + if (fieldType == MemoryFieldType::CodePointer) + newHexValue = QString::asprintf("0x%016llX", pointerValue); + else + newHexValue = QString::asprintf("0x%016llX", pointerValue); } + + valueHexField->setData(newHexValue, Qt::DisplayRole); valueHexField->setData(pointertmp, gsRoleRawValue); return true; } @@ -604,40 +606,6 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional switch (fieldType) { case MemoryFieldType::CodePointer: - { - if (pointerUpdate) - { - QString newValue; - if (newPointer == 0) - newValue = "nullptr"; - else if (valueMemoryOffset == 0) // trick to not use Script::Memory::IsValidPtr(newPointer) again - newValue = "bad ptr"; - else - newValue = QString::asprintf("0x%016llX", newPointer); - - itemValue->setData(newValue, Qt::DisplayRole); - itemValueHex->setData(newValue, Qt::DisplayRole); - } - - if (comparisonActive) - { - if (comparisonPointerUpdate) - { - QString newComparisonValue; - if (newComparisonPointer == 0) - newComparisonValue = "nullptr"; - else if (valueComparisonMemoryOffset == 0) - newComparisonValue = "bad ptr"; - else - newComparisonValue = QString::asprintf("0x%016llX", newComparisonPointer); - - itemComparisonValue->setData(newComparisonValue, Qt::DisplayRole); - itemComparisonValueHex->setData(newComparisonValue, Qt::DisplayRole); - } - itemComparisonValue->setBackground(itemComparisonValueHex->background()); - } - break; - } case MemoryFieldType::DataPointer: { if (pointerUpdate) @@ -676,7 +644,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional if (comparisonActive) { std::optional comparisonValue; - comparisonValue = updateField(itemField, valueComparisonMemoryOffset, itemComparisonValue, "%d", itemComparisonValueHex, isPointer, "0x%02X", false, false, highlightColor); + comparisonValue = updateField(itemField, valueComparisonMemoryOffset, itemComparisonValue, "%u", itemComparisonValueHex, isPointer, "0x%02X", false, false, highlightColor); itemComparisonValue->setBackground(value != comparisonValue ? comparisonDifferenceColor : Qt::transparent); if (isPointer == false) @@ -708,7 +676,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional if (comparisonActive) { std::optional comparisonValue; - comparisonValue = updateField(itemField, valueComparisonMemoryOffset, itemComparisonValue, "%d", itemComparisonValueHex, isPointer, "0x%04X", false, false, highlightColor); + comparisonValue = updateField(itemField, valueComparisonMemoryOffset, itemComparisonValue, "%u", itemComparisonValueHex, isPointer, "0x%04X", false, false, highlightColor); itemComparisonValue->setBackground(value != comparisonValue ? comparisonDifferenceColor : Qt::transparent); if (isPointer == false) @@ -735,12 +703,12 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional case MemoryFieldType::UnsignedDword: { std::optional value; - value = updateField(itemField, valueMemoryOffset, itemValue, "%u", itemValueHex, isPointer, "0x%08X", true, !pointerUpdate, highlightColor); + value = updateField(itemField, valueMemoryOffset, itemValue, "%lu", itemValueHex, isPointer, "0x%08X", true, !pointerUpdate, highlightColor); if (comparisonActive) { std::optional comparisonValue; - comparisonValue = updateField(itemField, valueComparisonMemoryOffset, itemComparisonValue, "%u", itemComparisonValueHex, isPointer, "0x%08X", false, false, highlightColor); + comparisonValue = updateField(itemField, valueComparisonMemoryOffset, itemComparisonValue, "%lu", itemComparisonValueHex, isPointer, "0x%08X", false, false, highlightColor); itemComparisonValue->setBackground(value != comparisonValue ? comparisonDifferenceColor : Qt::transparent); if (isPointer == false) @@ -819,14 +787,14 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional std::optional value; value = updateField(itemField, valueMemoryOffset, itemValue, nullptr, itemValueHex, isPointer, "0x%02X", true, !pointerUpdate, highlightColor); if (value.has_value()) - itemValue->setData(value.value() ? "True" : "False", Qt::DisplayRole); // maybe color them green/red as well? + itemValue->setData(value.value() ? "True" : "False", Qt::DisplayRole); if (comparisonActive) { std::optional comparisonValue; comparisonValue = updateField(itemField, valueComparisonMemoryOffset, itemComparisonValue, nullptr, itemComparisonValueHex, isPointer, "0x%02X", false, false, highlightColor); if (comparisonValue.has_value()) - itemComparisonValue->setData(comparisonValue.value() ? "True" : "False", Qt::DisplayRole); + itemComparisonValue->setData(comparisonValue.value() ? "True" : "False", Qt::DisplayRole); itemComparisonValue->setBackground(value != comparisonValue ? comparisonDifferenceColor : Qt::transparent); if (isPointer == false) @@ -925,14 +893,12 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional auto parentIndex = idx.parent(); return mod->data(mod->index(idx.row(), col, parentIndex), role); }; - + // [Known Issue]: null memory address is not handled auto flagIndex = itemField->data(gsRoleFlagIndex).toUInt(); uint mask = (1 << (flagIndex - 1)); - auto flagRef = qvariant_cast(itemField->parent()->data(gsRoleRefName)); - auto flagName = Configuration::get()->flagTitle(flagRef, flagIndex); auto value = getDataFrom(itemField->parent()->index(), gsColValue, gsRoleRawValue).toUInt(); - auto flagSet = ((value & mask) == mask); + auto flagSet = (value & mask) == mask; if (itemValue->data(gsRoleRawValue).toBool() != flagSet) { itemValue->setData(flagSet, gsRoleRawValue); @@ -941,19 +907,17 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional else itemField->setBackground(Qt::transparent); - auto flagTitle = QString::fromStdString(flagName.empty() ? Configuration::get()->flagTitle("unknown", flagIndex) : flagName); // TODO: don't show empty unless it was chosen in settings - // TODO: would love to instead get the names and save them in addMemoryField and then just use itemValue->setForeground or itemValue->setData(Qt::TextColorRole) for the color - // but it doesn't work with HTML delagate, and i don't know how to edit it to make it work - auto caption = QString("%2").arg(flagSet ? "green" : "red", flagTitle); - itemValue->setData(caption, Qt::DisplayRole); + itemValue->setData(flagSet ? QColor("green") : QColor("red"), Qt::TextColorRole); - auto comparisonValue = getDataFrom(itemField->parent()->index(), gsColComparisonValue, gsRoleRawValue).toUInt(); - auto comparisonFlagSet = ((comparisonValue & mask) == mask); - auto comparisonTitle = QString("%2").arg(comparisonFlagSet ? "green" : "red", flagTitle); - itemComparisonValue->setData(comparisonTitle, Qt::DisplayRole); + if (comparisonActive) + { + auto comparisonValue = getDataFrom(itemField->parent()->index(), gsColComparisonValue, gsRoleRawValue).toUInt(); + auto comparisonFlagSet = (comparisonValue & mask) == mask; + itemComparisonValue->setData(comparisonFlagSet ? QColor("green") : QColor("red"), Qt::TextColorRole); - itemComparisonValue->setBackground(flagSet != comparisonFlagSet ? comparisonDifferenceColor : Qt::transparent); - itemComparisonValueHex->setBackground(flagSet != comparisonFlagSet ? comparisonDifferenceColor : Qt::transparent); + itemComparisonValue->setBackground(flagSet != comparisonFlagSet ? comparisonDifferenceColor : Qt::transparent); + itemComparisonValueHex->setBackground(flagSet != comparisonFlagSet ? comparisonDifferenceColor : Qt::transparent); + } break; } case MemoryFieldType::State8: @@ -1048,7 +1012,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional std::optional value; value = updateField(itemField, valueMemoryOffset, itemValue, nullptr, itemValueHex, isPointer, "0x%04X", true, !pointerUpdate, highlightColor); if (value.has_value()) - itemValue->setData(QString("'%1' (%2)").arg(QChar(value.value())).arg(value.value()), Qt::DisplayRole); + itemValue->setData(QString("'%1' (%2)").arg(QChar(value.value())).arg(value.value()).toHtmlEscaped(), Qt::DisplayRole); if (comparisonActive) { @@ -1056,7 +1020,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional comparisonValue = updateField(itemField, valueComparisonMemoryOffset, itemComparisonValue, nullptr, itemComparisonValueHex, isPointer, "0x%04X", false, false, highlightColor); if (comparisonValue.has_value()) - itemComparisonValue->setData(QString("'%1' (%2)").arg(QChar(comparisonValue.value())).arg(comparisonValue.value()), Qt::DisplayRole); + itemComparisonValue->setData(QString("'%1' (%2)").arg(QChar(comparisonValue.value())).arg(comparisonValue.value()).toHtmlEscaped(), Qt::DisplayRole); itemComparisonValue->setBackground(value != comparisonValue ? comparisonDifferenceColor : Qt::transparent); if (isPointer == false) @@ -1072,18 +1036,18 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional if (valueMemoryOffset == 0) { itemField->setBackground(Qt::transparent); - itemValue->setData("", Qt::DisplayRole); + itemValue->setData({}, Qt::DisplayRole); if (!isPointer) - itemValueHex->setData("", Qt::DisplayRole); + itemValueHex->setData({}, Qt::DisplayRole); } else { Script::Memory::Read(valueMemoryOffset, buffer, size, nullptr); auto buffer_w = reinterpret_cast(buffer); - auto valueString = QString::fromUtf16(buffer_w); + auto valueString = ('\"' + QString::fromUtf16(buffer_w) + '\"').toHtmlEscaped(); - QString valueOld = itemValue->data(Qt::DisplayRole).toString(); // no need for gsRoleRawValue - if (valueString != valueOld) + auto valueOld = itemValue->data(Qt::DisplayRole); // no need for gsRoleRawValue + if (valueOld.isNull() || valueString != valueOld.toString()) { itemField->setBackground(highlightColor); itemValue->setData(valueString, Qt::DisplayRole); @@ -1091,7 +1055,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional { std::stringstream ss; ss << "0x" << std::hex << std::setfill('0'); - for (int i = 0; i < std::min(size / 2, 10ull) && buffer_w[i] != 0; ++i) + for (int i = 0; i < std::min(size / 2, 10ull); ++i) ss << std::setw(4) << static_cast(buffer_w[i]); itemValueHex->setData(QString::fromStdString(ss.str()), Qt::DisplayRole); @@ -1104,18 +1068,18 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional { if (valueComparisonMemoryOffset == 0) { - itemComparisonValue->setData("", Qt::DisplayRole); + itemComparisonValue->setData({}, Qt::DisplayRole); if (!isPointer) - itemComparisonValueHex->setData("", Qt::DisplayRole); + itemComparisonValueHex->setData({}, Qt::DisplayRole); } else { Script::Memory::Read(valueComparisonMemoryOffset, buffer, size, nullptr); auto buffer_w = reinterpret_cast(buffer); - auto valueString = QString::fromUtf16(buffer_w); + auto valueString = ('\"' + QString::fromUtf16(buffer_w) + '\"').toHtmlEscaped(); - QString valueOld = itemComparisonValue->data(Qt::DisplayRole).toString(); // no need for gsRoleRawValue - if (valueString != valueOld) + auto valueOld = itemComparisonValue->data(Qt::DisplayRole); // no need for gsRoleRawValue + if (valueOld.isNull() || valueString != valueOld.toString()) { itemField->setBackground(highlightColor); itemComparisonValue->setData(valueString, Qt::DisplayRole); @@ -1142,17 +1106,17 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional if (valueMemoryOffset == 0) { itemField->setBackground(Qt::transparent); - itemValue->setData("", Qt::DisplayRole); + itemValue->setData({}, Qt::DisplayRole); if (!isPointer) - itemValueHex->setData("", Qt::DisplayRole); + itemValueHex->setData({}, Qt::DisplayRole); } else { Script::Memory::Read(valueMemoryOffset, buffer, size, nullptr); - auto valueString = QString::fromUtf8(buffer); + auto valueString = ('\"' + QString::fromUtf8(buffer) + '\"').toHtmlEscaped(); - QString valueOld = itemValue->data(Qt::DisplayRole).toString(); // no need for gsRoleRawValue - if (valueString != valueOld) + auto valueOld = itemValue->data(Qt::DisplayRole); // no need for gsRoleRawValue + if (valueOld.isNull() || valueString != valueOld.toString()) { itemField->setBackground(highlightColor); itemValue->setData(valueString, Qt::DisplayRole); @@ -1160,7 +1124,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional { std::stringstream ss; ss << "0x" << std::hex << std::setfill('0'); - for (int i = 0; i < std::min(size, 10ull) && buffer[i] != 0; ++i) + for (int i = 0; i < std::min(size, 10ull); ++i) ss << std::setw(2) << static_cast(buffer[i]); itemValueHex->setData(QString::fromStdString(ss.str()), Qt::DisplayRole); @@ -1173,17 +1137,17 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional { if (valueComparisonMemoryOffset == 0) { - itemComparisonValue->setData("", Qt::DisplayRole); + itemComparisonValue->setData({}, Qt::DisplayRole); if (!isPointer) - itemComparisonValueHex->setData("", Qt::DisplayRole); + itemComparisonValueHex->setData({}, Qt::DisplayRole); } else { Script::Memory::Read(valueComparisonMemoryOffset, buffer, size, nullptr); - auto valueString = QString::fromUtf8(buffer); + auto valueString = ('\"' + QString::fromUtf8(buffer) + '\"').toHtmlEscaped(); - QString valueOld = itemComparisonValue->data(Qt::DisplayRole).toString(); // no need for gsRoleRawValue - if (valueString != valueOld) + auto valueOld = itemComparisonValue->data(Qt::DisplayRole); // no need for gsRoleRawValue + if (valueOld.isNull() || valueString != valueOld.toString()) { itemField->setBackground(highlightColor); itemComparisonValue->setData(valueString, Qt::DisplayRole); @@ -1191,7 +1155,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional { std::stringstream ss; ss << "0x" << std::hex << std::setfill('0'); - for (int i = 0; i < std::min(size, 10ull) && buffer[i] != 0; ++i) + for (int i = 0; i < std::min(size, 10ull); ++i) ss << std::setw(2) << static_cast(buffer[i]); itemComparisonValueHex->setData(QString::fromStdString(ss.str()), Qt::DisplayRole); @@ -1244,6 +1208,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional { if (value.value() < 0) { + // TODO: write better explanation itemValue->setData(QString::asprintf("%d (dynamically applied in ThemeInfo->get_dynamic_floor_texture_id())", value.value()), Qt::DisplayRole); } else @@ -1282,7 +1247,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional std::optional value; value = updateField(itemField, valueMemoryOffset, itemValue, nullptr, itemValueHex, isPointer, "0x%08X", true, !pointerUpdate, highlightColor); if (value.has_value()) - itemValue->setData(QString("%1: %2").arg(value.value()).arg(Spelunky2::get()->get_StringsTable().stringForIndex(value.value())), Qt::DisplayRole); + itemValue->setData(QString("%1: %2").arg(value.value()).arg(Spelunky2::get()->get_StringsTable().stringForIndex(value.value())).toHtmlEscaped(), Qt::DisplayRole); if (comparisonActive) { @@ -1291,7 +1256,8 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional updateField(itemField, valueComparisonMemoryOffset, itemComparisonValue, nullptr, itemComparisonValueHex, isPointer, "0x%08X", false, false, highlightColor); if (comparisonValue.has_value()) { - itemComparisonValue->setData(QString("%1: %2").arg(comparisonValue.value()).arg(Spelunky2::get()->get_StringsTable().stringForIndex(comparisonValue.value())), Qt::DisplayRole); + itemComparisonValue->setData(QString("%1: %2").arg(comparisonValue.value()).arg(Spelunky2::get()->get_StringsTable().stringForIndex(comparisonValue.value())).toHtmlEscaped(), + Qt::DisplayRole); } itemComparisonValue->setBackground(value != comparisonValue ? comparisonDifferenceColor : Qt::transparent); @@ -1420,7 +1386,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional itemValue->setData(itemValueHex->data(Qt::DisplayRole), Qt::DisplayRole); else { - auto id = Script::Memory::ReadDword(valueMemoryOffset + 20); // TODO hex offset + auto id = Script::Memory::ReadDword(valueMemoryOffset + 0x14); auto entityName = Configuration::get()->entityList().nameForID(id); itemValue->setData(QString::asprintf("EntityDB %d %s", id, entityName.c_str()), Qt::DisplayRole); } @@ -1569,8 +1535,8 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional } else { - std::string str = ReadConstString(valueMemoryOffset); - itemValue->setData(QString::fromStdString(str), Qt::DisplayRole); + std::string str = '\"' + ReadConstString(valueMemoryOffset) + '\"'; + itemValue->setData(QString::fromStdString(str).toHtmlEscaped(), Qt::DisplayRole); } if (comparisonActive) @@ -1581,8 +1547,8 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional } else { - std::string comparisonStr = ReadConstString(valueComparisonMemoryOffset); - itemComparisonValue->setData(QString::fromStdString(comparisonStr), Qt::DisplayRole); + std::string comparisonStr = '\"' + ReadConstString(valueComparisonMemoryOffset) + '\"'; + itemComparisonValue->setData(QString::fromStdString(comparisonStr).toHtmlEscaped(), Qt::DisplayRole); } // pointer compare itemComparisonValue->setBackground(itemComparisonValueHex->background()); @@ -1596,20 +1562,20 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional if (valueMemoryOffset == 0) { itemField->setBackground(Qt::transparent); - itemValue->setData("", Qt::DisplayRole); + itemValue->setData({}, Qt::DisplayRole); if (!isPointer) - itemValueHex->setData("", Qt::DisplayRole); + itemValueHex->setData({}, Qt::DisplayRole); } else { StdString string{valueMemoryOffset}; - // i don't think we will have pointer to std::string, but note just in case: this would override the pointer value in hex + // [Known Issue]: i don't think we will have pointer to std::string, but note just in case: this would override the pointer value in hex auto ptr = string.string_ptr(); itemValueHex->setData(QString::asprintf("0x%016llX", ptr), Qt::DisplayRole); itemValueHex->setData(ptr, gsRoleRawValue); - stringValue = string.get_string(); - auto displayValue = QString::fromStdString(stringValue.value()); + stringValue = '\"' + string.get_string() + '\"'; + auto displayValue = QString::fromStdString(stringValue.value()).toHtmlEscaped(); itemField->setBackground(itemValue->data(Qt::DisplayRole).toString() == displayValue ? Qt::transparent : highlightColor); itemValue->setData(displayValue, Qt::DisplayRole); } @@ -1618,15 +1584,15 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional { if (valueComparisonMemoryOffset == 0) { - itemComparisonValue->setData("", Qt::DisplayRole); + itemComparisonValue->setData({}, Qt::DisplayRole); if (!isPointer) - itemComparisonValueHex->setData("", Qt::DisplayRole); + itemComparisonValueHex->setData({}, Qt::DisplayRole); } else { StdString comparisonString{valueComparisonMemoryOffset}; - comparisonStringValue = comparisonString.get_string(); - itemComparisonValue->setData(QString::fromStdString(comparisonStringValue.value()), Qt::DisplayRole); + comparisonStringValue = '\"' + comparisonString.get_string() + '\"'; + itemComparisonValue->setData(QString::fromStdString(comparisonStringValue.value()).toHtmlEscaped(), Qt::DisplayRole); auto ptr = comparisonString.string_ptr(); itemComparisonValueHex->setData(QString::asprintf("0x%016llX", ptr), Qt::DisplayRole); @@ -1640,25 +1606,24 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional } if (shouldUpdateChildren) { + std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); + std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); for (uint8_t x = 0; x < itemField->rowCount(); ++x) - { - std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); - std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); updateRow(x, addr, comparisonAddr, itemField); - } } break; } case MemoryFieldType::StdWstring: { - std::optional stringValue; - std::optional comparisonStringValue; + constexpr ushort quotationMark = static_cast(u'\"'); + std::optional> stringValue; + std::optional> comparisonStringValue; if (valueMemoryOffset == 0) { itemField->setBackground(Qt::transparent); - itemValue->setData("", Qt::DisplayRole); + itemValue->setData({}, Qt::DisplayRole); if (!isPointer) - itemValueHex->setData("", Qt::DisplayRole); + itemValueHex->setData({}, Qt::DisplayRole); } else { @@ -1668,25 +1633,26 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional itemValueHex->setData(QString::asprintf("0x%016llX", ptr), Qt::DisplayRole); itemValueHex->setData(ptr, gsRoleRawValue); - stringValue = string.get_string(); - // auto displayValue = QString::fromStdU16String(stringValue.value()); - // itemField->setBackground(itemValue->data(Qt::DisplayRole).toString() == displayValue ? Qt::transparent : highlightColor); - // itemValue->setData(displayValue, Qt::DisplayRole); + stringValue = quotationMark + string.get_string() + quotationMark; + auto displayValue = QString::fromUtf16(stringValue->c_str(), static_cast(stringValue->size())).toHtmlEscaped(); + itemField->setBackground(itemValue->data(Qt::DisplayRole).toString() == displayValue ? Qt::transparent : highlightColor); + itemValue->setData(displayValue, Qt::DisplayRole); } if (comparisonActive) { if (valueComparisonMemoryOffset == 0) { - itemComparisonValue->setData("", Qt::DisplayRole); + itemComparisonValue->setData({}, Qt::DisplayRole); if (!isPointer) - itemComparisonValueHex->setData("", Qt::DisplayRole); + itemComparisonValueHex->setData({}, Qt::DisplayRole); } else { StdWstring comparisonString{valueComparisonMemoryOffset}; - comparisonStringValue = comparisonString.get_string(); - // itemComparisonValue->setData(QString::fromStdU16String(comparisonStringValue.value()), Qt::DisplayRole); + comparisonStringValue = quotationMark + comparisonString.get_string() + quotationMark; + auto displayValue = QString::fromUtf16(comparisonStringValue->data(), static_cast(comparisonStringValue->size())).toHtmlEscaped(); + itemComparisonValue->setData(displayValue, Qt::DisplayRole); auto ptr = comparisonString.string_ptr(); itemComparisonValueHex->setData(QString::asprintf("0x%016llX", ptr), Qt::DisplayRole); @@ -1699,16 +1665,14 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional } if (shouldUpdateChildren) { + std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); + std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); for (uint8_t x = 0; x < itemField->rowCount(); ++x) - { - std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); - std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); updateRow(x, addr, comparisonAddr, itemField); - } } break; } - case MemoryFieldType::ThemeInfoName: + case MemoryFieldType::ThemeInfoPointer: case MemoryFieldType::UndeterminedThemeInfoPointer: { if (valueMemoryOffset == 0) @@ -1728,12 +1692,10 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional if (shouldUpdateChildren && fieldType == MemoryFieldType::UndeterminedThemeInfoPointer) { + std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); + std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); for (uint8_t x = 0; x < itemField->rowCount(); ++x) - { - std::optional addr = pointerUpdate ? std::optional(valueMemoryOffset) : std::nullopt; - std::optional comparisonAddr = comparisonPointerUpdate ? std::optional(valueComparisonMemoryOffset) : std::nullopt; - updateRow(x, valueMemoryOffset, valueComparisonMemoryOffset, itemField); - } + updateRow(x, addr, comparisonAddr, itemField); } break; } @@ -1749,7 +1711,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional { // we can't show comparison version since it's a tab, not a new window // TODO: maybe add comparison in the show rooms tab? - itemComparisonValue->setData("", Qt::DisplayRole); + itemComparisonValue->setData({}, Qt::DisplayRole); itemComparisonValue->setData(0, gsRoleMemoryAddress); itemComparisonValue->setBackground(itemComparisonValueHex->background()); @@ -1842,12 +1804,10 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional } if (shouldUpdateChildren) { + std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); + std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); for (uint8_t x = 0; x < itemField->rowCount(); ++x) - { - std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); - std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); updateRow(x, addr, comparisonAddr, itemField); - } } break; } @@ -1879,12 +1839,10 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional } if (shouldUpdateChildren) { + std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); + std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); for (uint8_t x = 0; x < itemField->rowCount(); ++x) - { - std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); - std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); updateRow(x, addr, comparisonAddr, itemField); - } } break; } @@ -1898,7 +1856,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional // can't be a pointer, nothing to do here if (shouldUpdateChildren) { - for (uint8_t x = 0; x < itemField->rowCount(); ++x) + for (int x = 0; x < itemField->rowCount(); ++x) { updateRow(x, newAddr, newAddrComparison, itemField); } @@ -1918,12 +1876,10 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional } if (shouldUpdateChildren) { - for (uint8_t x = 0; x < itemField->rowCount(); ++x) - { - std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); - std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); + std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); + std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); + for (int x = 0; x < itemField->rowCount(); ++x) updateRow(x, addr, comparisonAddr, itemField); - } } break; } @@ -1931,7 +1887,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional { if (shouldUpdateChildren) { - for (uint8_t x = 0; x < itemField->rowCount(); ++x) + for (int x = 0; x < itemField->rowCount(); ++x) updateRow(x, newAddr, newAddrComparison, itemField); } break; @@ -1942,7 +1898,7 @@ void S2Plugin::TreeViewMemoryFields::updateRow(int row, std::optional dprintf("WARNING: type %d not handled in TreeViewMemoryFields::updateRow ('%s' row: %d)\n", fieldType, itemField->data(gsRoleUID).toString().toStdString().c_str(), row); if (shouldUpdateChildren) { - for (uint8_t x = 0; x < itemField->rowCount(); ++x) + for (int x = 0; x < itemField->rowCount(); ++x) { std::optional addr = pointerUpdate ? valueMemoryOffset : (isPointer ? std::nullopt : newAddr); std::optional comparisonAddr = comparisonPointerUpdate ? valueComparisonMemoryOffset : (isPointer ? std::nullopt : newAddrComparison); @@ -2288,11 +2244,6 @@ void S2Plugin::TreeViewMemoryFields::clear() mModel->clear(); } -void S2Plugin::TreeViewMemoryFields::setEnableChangeHighlighting(bool b) noexcept -{ - mEnableChangeHighlighting = b; -} - void S2Plugin::TreeViewMemoryFields::dragEnterEvent(QDragEnterEvent* event) { if (event->mimeData()->hasFormat("spelunky/entityoffset")) @@ -2387,7 +2338,6 @@ static void labelChildren(QStandardItem* parrent, std::string_view prefix) name.reserve(prefix.length() + 1u + qstr_name.length()); name.append(prefix); name += '.'; - // TODO: use 'toUtf8' instead of toStdString to skip creating new object - struggling with stupid linker error name.append(qstr_name.toStdString()); } diff --git a/src/Views/ViewThreads.cpp b/src/Views/ViewThreads.cpp index 384510b6..a5a0da4c 100644 --- a/src/Views/ViewThreads.cpp +++ b/src/Views/ViewThreads.cpp @@ -141,12 +141,17 @@ void S2Plugin::ViewThreads::cellClicked(int row, int column) auto clickedItem = mMainTable->item(row, column); if (column == gsColTEBAddress) { - GuiDumpAt(clickedItem->data(gsRoleMemoryAddress).toULongLong()); - GuiShowCpu(); + auto addr = clickedItem->data(gsRoleMemoryAddress).toULongLong(); + if (addr != 0) + { + GuiDumpAt(addr); + GuiShowCpu(); + } } else if (column == gsColStateAddress) { auto statePtr = clickedItem->data(gsRoleMemoryAddress).toULongLong(); - getToolbar()->showState(statePtr); + if (statePtr != 0) + getToolbar()->showState(statePtr); } }