2022-03-04 19:06:29 +01:00
# include "content/views/view_about.hpp"
2022-02-01 18:09:40 +01:00
2023-11-05 19:57:29 +01:00
# include <hex/api_urls.hpp>
2022-02-01 18:09:40 +01:00
# include <hex/api/content_registry.hpp>
2023-08-06 21:33:15 +02:00
# include <hex/api/achievement_manager.hpp>
2023-12-18 11:21:33 +01:00
# include <hex/api/plugin_manager.hpp>
2020-11-14 21:16:03 +01:00
2021-09-06 20:35:38 +02:00
# include <hex/helpers/fmt.hpp>
2022-03-04 11:36:37 +01:00
# include <hex/helpers/fs.hpp>
2022-02-01 18:09:40 +01:00
# include <hex/helpers/utils.hpp>
2023-11-05 19:57:29 +01:00
# include <hex/helpers/http_requests.hpp>
2021-09-06 20:35:38 +02:00
2023-05-12 08:38:32 +02:00
# include <content/popups/popup_docs_question.hpp>
2024-01-28 22:14:59 +01:00
# include <fonts/codicons_font.h>
2022-01-22 23:11:28 +01:00
# include <romfs/romfs.hpp>
2023-11-05 19:57:29 +01:00
# include <wolv/utils/string.hpp>
2022-01-22 23:11:28 +01:00
2023-12-31 13:53:44 +01:00
# include <string>
2023-12-17 20:33:17 +01:00
2021-12-07 22:47:41 +01:00
namespace hex : : plugin : : builtin {
2020-11-14 21:16:03 +01:00
2023-12-17 20:33:17 +01:00
class PopupEE : public Popup < PopupEE > {
public :
PopupEE ( ) : Popup ( " Se " /* Not going to */ " cr " /* make it that easy */ " et " ) {
}
void drawContent ( ) override {
ImGuiIO & io = ImGui : : GetIO ( ) ;
ImVec2 size = scaled ( { 320 , 180 } ) ;
ImGui : : InvisibleButton ( " canvas " , size ) ;
ImVec2 p0 = ImGui : : GetItemRectMin ( ) ;
ImVec2 p1 = ImGui : : GetItemRectMax ( ) ;
ImDrawList * drawList = ImGui : : GetWindowDrawList ( ) ;
drawList - > PushClipRect ( p0 , p1 ) ;
ImVec4 mouseData ;
mouseData . x = ( io . MousePos . x - p0 . x ) / size . x ;
mouseData . y = ( io . MousePos . y - p0 . y ) / size . y ;
mouseData . z = io . MouseDownDuration [ 0 ] ;
mouseData . w = io . MouseDownDuration [ 1 ] ;
fx ( drawList , p0 , p1 , size , mouseData , float ( ImGui : : GetTime ( ) ) ) ;
}
void fx ( ImDrawList * drawList , ImVec2 startPos , ImVec2 endPos , ImVec2 , ImVec4 , float t ) {
const float CircleRadius = 5 _scaled ;
const float Gap = 1 _scaled ;
2023-12-18 08:58:00 +01:00
constexpr static auto func = [ ] ( i32 x , i32 y , float t ) {
2023-12-17 20:33:17 +01:00
return std : : sin ( t - std : : sqrt ( std : : pow ( ( x - 14 ) , 2 ) + std : : pow ( ( y - 8 ) , 2 ) ) ) ;
} ;
float x = startPos . x + CircleRadius + Gap ;
u32 ix = 0 ;
while ( x < endPos . x ) {
float y = startPos . y + CircleRadius + Gap ;
u32 iy = 0 ;
while ( y < endPos . y ) {
const float result = func ( ix , iy , t ) ;
const float radius = CircleRadius * std : : abs ( result ) ;
const auto color = result < 0 ? ImColor ( 0xFF , 0 , 0 , 0xFF ) : ImColor ( 0xFF , 0xFF , 0xFF , 0xFF ) ;
drawList - > AddCircleFilled ( ImVec2 ( x , y ) , radius , color ) ;
y + = CircleRadius * 2 + Gap ;
iy + = 1 ;
}
x + = CircleRadius * 2 + Gap ;
ix + = 1 ;
}
}
} ;
2023-11-21 13:47:50 +01:00
ViewAbout : : ViewAbout ( ) : View : : Modal ( " hex.builtin.view.help.about.name " ) {
2023-08-26 12:54:52 +02:00
// Add "About" menu item to the help menu
2024-01-08 21:51:48 +01:00
ContentRegistry : : Interface : : addMenuItem ( { " hex.builtin.menu.help " , " hex.builtin.view.help.about.name " } , ICON_VS_INFO , 1000 , Shortcut : : None , [ this ] {
2023-03-20 13:11:43 +01:00
this - > getWindowOpenState ( ) = true ;
} ) ;
2023-05-12 08:38:32 +02:00
ContentRegistry : : Interface : : addMenuItemSeparator ( { " hex.builtin.menu.help " } , 2000 ) ;
2023-12-31 13:53:44 +01:00
ContentRegistry : : Interface : : addMenuItemSubMenu ( { " hex.builtin.menu.help " } , 3000 , [ ] {
static std : : string content ;
if ( ImGui : : InputTextWithHint ( " ##search " , " hex.builtin.view.help.documentation_search " _lang , content , ImGuiInputTextFlags_AutoSelectAll | ImGuiInputTextFlags_EscapeClearsAll | ImGuiInputTextFlags_EnterReturnsTrue ) ) {
PopupDocsQuestion : : open ( content ) ;
content . clear ( ) ;
ImGui : : CloseCurrentPopup ( ) ;
}
2022-01-23 02:28:38 +01:00
} ) ;
2023-05-12 08:38:32 +02:00
2023-12-31 13:53:44 +01:00
ContentRegistry : : Interface : : addMenuItemSeparator ( { " hex.builtin.menu.help " } , 4000 ) ;
// Add documentation link to the help menu
2024-01-08 21:51:48 +01:00
ContentRegistry : : Interface : : addMenuItem ( { " hex.builtin.menu.help " , " hex.builtin.view.help.documentation " } , ICON_VS_BOOK , 5000 , Shortcut : : None , [ ] {
2023-12-31 13:53:44 +01:00
hex : : openWebpage ( " https://docs.werwolv.net/imhex " ) ;
AchievementManager : : unlockAchievement ( " hex.builtin.achievement.starting_out " , " hex.builtin.achievement.starting_out.docs.name " ) ;
2023-05-12 08:38:32 +02:00
} ) ;
2020-11-14 21:16:03 +01:00
}
2020-11-30 21:44:40 +01:00
2023-12-17 20:33:17 +01:00
void ViewAbout : : drawAboutMainPage ( ) {
2023-08-26 12:54:52 +02:00
// Draw main about table
2022-01-23 20:46:19 +01:00
if ( ImGui : : BeginTable ( " about_table " , 2 , ImGuiTableFlags_SizingFixedFit ) ) {
ImGui : : TableNextRow ( ) ;
ImGui : : TableNextColumn ( ) ;
2023-08-26 12:54:52 +02:00
// Draw the ImHex icon
2023-12-19 13:10:25 +01:00
if ( ! m_logoTexture . isValid ( ) )
2024-05-01 20:36:10 +02:00
m_logoTexture = ImGuiExt : : Texture : : fromImage ( romfs : : get ( " assets/common/logo.png " ) . span ( ) , ImGuiExt : : Texture : : Filter : : Linear ) ;
2023-08-26 23:43:35 +02:00
2023-12-19 13:10:25 +01:00
ImGui : : Image ( m_logoTexture , scaled ( { 100 , 100 } ) ) ;
2023-12-17 20:33:17 +01:00
if ( ImGui : : IsItemClicked ( ) ) {
2023-12-19 13:10:25 +01:00
m_clickCount + = 1 ;
2023-11-05 19:57:29 +01:00
}
2023-12-17 20:33:17 +01:00
2023-12-19 13:10:25 +01:00
if ( m_clickCount > = ( 2 * 3 + 4 ) ) {
2023-12-17 20:33:17 +01:00
this - > getWindowOpenState ( ) = false ;
PopupEE : : open ( ) ;
2023-12-19 13:10:25 +01:00
m_clickCount = 0 ;
2023-12-17 20:33:17 +01:00
}
2022-01-23 20:46:19 +01:00
ImGui : : TableNextColumn ( ) ;
2021-08-25 13:54:59 -04:00
2024-02-11 00:35:10 +01:00
ImGuiExt : : BeginSubWindow ( " Build Information " , ImVec2 ( 450 _scaled , 0 ) , ImGuiChildFlags_AutoResizeX | ImGuiChildFlags_AutoResizeY ) ;
2023-11-25 00:43:03 +01:00
{
if ( ImGui : : BeginTable ( " Information " , 1 , ImGuiTableFlags_RowBg | ImGuiTableFlags_BordersInner ) ) {
2024-02-11 00:35:10 +01:00
ImGui : : Indent ( ) ;
2021-12-12 13:35:07 +01:00
2024-02-11 00:35:10 +01:00
ImGui : : TableNextRow ( ) ;
2023-11-25 00:43:03 +01:00
ImGui : : TableNextColumn ( ) ;
{
// Draw basic information about ImHex and its version
2023-12-11 23:05:35 +01:00
ImGuiExt : : TextFormatted ( " ImHex Hex Editor v{} by WerWolv " , ImHexApi : : System : : getImHexVersion ( ) ) ;
2024-02-11 00:11:56 +01:00
ImGui : : Indent ( 25 _scaled ) ;
ImGuiExt : : TextFormatted ( " Powered by Dear ImGui v{} " , ImGui : : GetVersion ( ) ) ;
ImGui : : Unindent ( 25 _scaled ) ;
2023-12-11 23:05:35 +01:00
}
2023-08-26 12:54:52 +02:00
2023-12-11 23:05:35 +01:00
ImGui : : TableNextColumn ( ) ;
{
ImGuiExt : : TextFormatted ( " {} " , ICON_VS_SOURCE_CONTROL ) ;
ImGui : : SameLine ( 0 , 0 ) ;
2022-01-23 20:46:19 +01:00
2023-11-25 00:43:03 +01:00
// Draw a clickable link to the current commit
if ( ImGuiExt : : Hyperlink ( hex : : format ( " {0}@{1} " , ImHexApi : : System : : getCommitBranch ( ) , ImHexApi : : System : : getCommitHash ( ) ) . c_str ( ) ) )
hex : : openWebpage ( " https://github.com/WerWolv/ImHex/commit/ " + ImHexApi : : System : : getCommitHash ( true ) ) ;
}
2023-10-30 23:24:00 +01:00
2023-11-25 00:43:03 +01:00
ImGui : : TableNextColumn ( ) ;
{
// Draw the build date and time
ImGuiExt : : TextFormatted ( " Compiled on {} at {} " , __DATE__ , __TIME__ ) ;
}
2022-01-23 20:46:19 +01:00
2023-11-25 00:43:03 +01:00
ImGui : : TableNextColumn ( ) ;
{
// Draw the author of the current translation
ImGui : : TextUnformatted ( " hex.builtin.view.help.about.translator " _lang ) ;
}
2023-08-26 12:54:52 +02:00
2023-11-25 00:43:03 +01:00
ImGui : : TableNextColumn ( ) ;
{
// Draw information about the open-source nature of ImHex
ImGui : : TextUnformatted ( " hex.builtin.view.help.about.source " _lang ) ;
ImGui : : SameLine ( ) ;
2023-08-26 12:54:52 +02:00
2023-11-25 00:43:03 +01:00
// Draw a clickable link to the GitHub repository
2024-02-11 00:35:10 +01:00
if ( ImGuiExt : : Hyperlink ( ICON_VS_LOGO_GITHUB " " " WerWolv/ImHex " ) )
2023-11-25 00:43:03 +01:00
hex : : openWebpage ( " https://github.com/WerWolv/ImHex " ) ;
}
2024-02-11 00:35:10 +01:00
ImGui : : Unindent ( ) ;
2023-11-25 00:43:03 +01:00
ImGui : : EndTable ( ) ;
}
}
ImGuiExt : : EndSubWindow ( ) ;
2022-01-23 20:46:19 +01:00
ImGui : : EndTable ( ) ;
}
2020-11-30 21:44:40 +01:00
2023-08-26 12:54:52 +02:00
// Draw donation links
2023-11-25 00:43:03 +01:00
ImGuiExt : : Header ( " hex.builtin.view.help.about.donations " _lang ) ;
2021-12-12 13:35:07 +01:00
2024-02-11 00:35:10 +01:00
if ( ImGui : : BeginChild ( " ##ThanksWrapper " , ImVec2 ( ImGui : : GetContentRegionAvail ( ) . x , ImGui : : GetTextLineHeightWithSpacing ( ) * 3 ) ) ) {
ImGui : : PushTextWrapPos ( ImGui : : GetContentRegionAvail ( ) . x * 0.8F ) ;
ImGuiExt : : TextFormattedCentered ( " {} " , static_cast < const char * > ( " hex.builtin.view.help.about.thanks " _lang ) ) ;
ImGui : : PopTextWrapPos ( ) ;
}
ImGui : : EndChild ( ) ;
2021-12-12 13:35:07 +01:00
ImGui : : NewLine ( ) ;
2021-01-23 00:46:50 +01:00
2023-11-25 00:43:03 +01:00
struct DonationPage {
ImGuiExt : : Texture texture ;
const char * link ;
} ;
static std : : array DonationPages = {
2024-05-01 20:36:10 +02:00
DonationPage { ImGuiExt : : Texture : : fromImage ( romfs : : get ( " assets/common/donation/paypal.png " ) . span < std : : byte > ( ) , ImGuiExt : : Texture : : Filter : : Linear ) , " https://werwolv.net/donate " } ,
DonationPage { ImGuiExt : : Texture : : fromImage ( romfs : : get ( " assets/common/donation/github.png " ) . span < std : : byte > ( ) , ImGuiExt : : Texture : : Filter : : Linear ) , " https://github.com/sponsors/WerWolv " } ,
DonationPage { ImGuiExt : : Texture : : fromImage ( romfs : : get ( " assets/common/donation/patreon.png " ) . span < std : : byte > ( ) , ImGuiExt : : Texture : : Filter : : Linear ) , " https://patreon.com/werwolv " } ,
2023-11-25 00:43:03 +01:00
} ;
2023-11-25 19:11:14 +01:00
if ( ImGui : : BeginTable ( " DonationLinks " , 5 , ImGuiTableFlags_SizingStretchSame ) ) {
2023-11-25 00:43:03 +01:00
ImGui : : TableNextRow ( ) ;
ImGui : : TableNextColumn ( ) ;
for ( const auto & page : DonationPages ) {
ImGui : : TableNextColumn ( ) ;
const auto size = page . texture . getSize ( ) / 1.5F ;
const auto startPos = ImGui : : GetCursorScreenPos ( ) ;
ImGui : : Image ( page . texture , page . texture . getSize ( ) / 1.5F ) ;
if ( ImGui : : IsItemHovered ( ) ) {
ImGui : : GetForegroundDrawList ( ) - > AddShadowCircle ( startPos + size / 2 , size . x / 2 , ImGui : : GetColorU32 ( ImGuiCol_Button ) , 100.0F , ImVec2 ( ) , ImDrawFlags_ShadowCutOutShapeBackground ) ;
}
if ( ImGui : : IsItemClicked ( ) ) {
hex : : openWebpage ( page . link ) ;
}
}
ImGui : : EndTable ( ) ;
2021-12-12 13:35:07 +01:00
}
2024-02-11 00:35:10 +01:00
ImGui : : NewLine ( ) ;
2021-12-12 13:35:07 +01:00
}
2021-01-23 00:46:50 +01:00
2024-03-02 11:57:30 +01:00
struct Contributor {
const char * name ;
const char * description ;
const char * link ;
bool mainContributor ;
} ;
2023-08-26 12:54:52 +02:00
2024-03-02 11:57:30 +01:00
static void drawContributorTable ( const char * title , const auto & contributors ) {
2023-11-25 00:43:03 +01:00
ImGui : : PushStyleVar ( ImGuiStyleVar_WindowPadding , ImVec2 ( ) ) ;
2024-03-02 11:57:30 +01:00
ImGuiExt : : BeginSubWindow ( title , ImVec2 ( ImGui : : GetContentRegionAvail ( ) . x , 0 ) , ImGuiChildFlags_AutoResizeX ) ;
2023-11-25 12:44:20 +01:00
ImGui : : PopStyleVar ( ) ;
2023-11-25 00:43:03 +01:00
{
2024-03-02 11:57:30 +01:00
if ( ImGui : : BeginTable ( title , 1 , ImGuiTableFlags_RowBg | ImGuiTableFlags_Borders ) ) {
for ( const auto & contributor : contributors ) {
2023-11-25 00:43:03 +01:00
ImGui : : TableNextRow ( ) ;
2023-11-25 13:15:21 +01:00
if ( contributor . mainContributor ) {
2023-11-25 13:42:51 +01:00
ImGui : : TableSetBgColor ( ImGuiTableBgTarget_RowBg0 , ImGui : : GetColorU32 ( ImGuiCol_PlotHistogram ) & 0x1FFFFFFF ) ;
2023-11-25 13:15:21 +01:00
}
2023-11-25 00:43:03 +01:00
ImGui : : TableNextColumn ( ) ;
2022-03-04 19:06:29 +01:00
2023-11-25 13:15:21 +01:00
if ( ImGuiExt : : Hyperlink ( contributor . name ) )
hex : : openWebpage ( contributor . link ) ;
2023-11-25 12:44:20 +01:00
2024-03-02 11:57:30 +01:00
if ( contributor . description [ 0 ] ! = ' \0 ' ) {
ImGui : : Indent ( ) ;
ImGui : : TextUnformatted ( contributor . description ) ;
ImGui : : Unindent ( ) ;
}
2023-11-25 00:43:03 +01:00
}
ImGui : : EndTable ( ) ;
}
}
ImGuiExt : : EndSubWindow ( ) ;
2021-12-12 13:35:07 +01:00
}
2021-01-23 00:46:50 +01:00
2024-03-02 11:57:30 +01:00
void ViewAbout : : drawContributorPage ( ) {
constexpr static std : : array Contributors = {
Contributor { " iTrooz " , " A huge amount of help maintaining ImHex and the CI " , " https://github.com/iTrooz " , true } ,
Contributor { " jumanji144 " , " A ton of help with the Pattern Language, API and usage stats " , " https://github.com/jumanji144 " , true } ,
2024-03-11 21:26:03 +01:00
Contributor { " AxCut " , " A ton of great pattern language improvements and help with the issue tracker " , " https://github.com/paxcut " , false } ,
2024-03-02 11:57:30 +01:00
Contributor { " Mary " , " Porting ImHex to macOS originally " , " https://github.com/marysaka " , false } ,
Contributor { " Roblabla " , " Adding the MSI Windows installer " , " https://github.com/roblabla " , false } ,
Contributor { " jam1garner " , " Adding support for Rust plugins " , " https://github.com/jam1garner " , false } ,
Contributor { " All other amazing contributors " , " Being part of the community, opening issues, PRs and donating " , " https://github.com/WerWolv/ImHex/graphs/contributors " , false }
} ;
constexpr static std : : array Testers = {
Contributor { " Nemoumbra " , " Breaking my code literal seconds after I push it " , " https://github.com/Nemoumbra " , true } ,
Contributor { " Berylskid " , " " , " https://github.com/Berylskid " , false } ,
Contributor { " Jan Polak " , " " , " https://github.com/polak-jan " , false } ,
Contributor { " Ken-Kaneki " , " " , " https://github.com/loneicewolf " , false } ,
Contributor { " Everybody who has reported issues " , " Helping me find bugs and improve the software " , " https://github.com/WerWolv/ImHex/issues " , false }
} ;
ImGuiExt : : TextFormattedWrapped ( " These amazing people have contributed some incredible things to ImHex in the past. \n Consider opening a PR on the Git Repository to take your place among them! " ) ;
ImGui : : NewLine ( ) ;
drawContributorTable ( " Contributors " , Contributors ) ;
ImGui : : NewLine ( ) ;
ImGuiExt : : TextFormattedWrapped ( " All of these great people made ImHex work much much smoother. \n Consider joining our Tester team to help making ImHex better for everyone! " ) ;
ImGui : : NewLine ( ) ;
drawContributorTable ( " Testers " , Testers ) ;
}
2022-03-04 19:06:29 +01:00
void ViewAbout : : drawLibraryCreditsPage ( ) {
2023-11-25 00:43:03 +01:00
struct Library {
const char * name ;
const char * author ;
const char * link ;
} ;
2022-03-04 19:06:29 +01:00
2023-11-25 00:43:03 +01:00
constexpr static std : : array ImGuiLibraries = {
Library { " ImGui " , " ocornut " , " https://github.com/ocornut/imgui " } ,
Library { " ImPlot " , " epezent " , " https://github.com/epezent/implot " } ,
Library { " imnodes " , " Nelarius " , " https://github.com/Nelarius/imnodes " } ,
Library { " ImGuiColorTextEdit " , " BalazsJako " , " https://github.com/BalazsJako/ImGuiColorTextEdit " } ,
} ;
2022-03-04 19:06:29 +01:00
2023-11-25 00:43:03 +01:00
constexpr static std : : array ExternalLibraries = {
Library { " PatternLanguage " , " WerWolv " , " https://github.com/WerWolv/PatternLanguage " } ,
Library { " libwolv " , " WerWolv " , " https://github.com/WerWolv/libwolv " } ,
Library { " libromfs " , " WerWolv " , " https://github.com/WerWolv/libromfs " } ,
} ;
2022-03-04 19:06:29 +01:00
2023-11-25 00:43:03 +01:00
constexpr static std : : array ThirdPartyLibraries = {
Library { " json " , " nlohmann " , " https://github.com/nlohmann/json " } ,
2024-03-25 21:02:55 +01:00
Library { " fmt " , " fmtlib " , " https://github.com/fmtlib/fmt " } ,
2023-11-25 00:43:03 +01:00
Library { " nativefiledialog-extended " , " btzy " , " https://github.com/btzy/nativefiledialog-extended " } ,
Library { " xdgpp " , " danyspin97 " , " https://sr.ht/~danyspin97/xdgpp " } ,
2024-03-25 21:02:55 +01:00
Library { " capstone " , " aquynh " , " https://github.com/aquynh/capstone " } ,
Library { " microtar " , " rxi " , " https://github.com/rxi/microtar " } ,
Library { " yara " , " VirusTotal " , " https://github.com/VirusTotal/yara " } ,
Library { " edlib " , " Martinsos " , " https://github.com/Martinsos/edlib " } ,
Library { " HashLibPlus " , " ron4fun " , " https://github.com/ron4fun/HashLibPlus " } ,
Library { " miniaudio " , " mackron " , " https://github.com/mackron/miniaudio " } ,
2023-11-25 00:43:03 +01:00
Library { " freetype " , " freetype " , " https://gitlab.freedesktop.org/freetype/freetype " } ,
Library { " mbedTLS " , " ARMmbed " , " https://github.com/ARMmbed/mbedtls " } ,
Library { " curl " , " curl " , " https://github.com/curl/curl " } ,
Library { " file " , " file " , " https://github.com/file/file " } ,
Library { " glfw " , " glfw " , " https://github.com/glfw/glfw " } ,
Library { " llvm " , " llvm-project " , " https://github.com/llvm/llvm-project " } ,
} ;
constexpr static auto drawTable = [ ] ( const char * category , const auto & libraries ) {
const auto width = ImGui : : GetContentRegionAvail ( ) . x ;
ImGuiExt : : BeginSubWindow ( category ) ;
{
for ( const auto & library : libraries ) {
ImGui : : PushStyleColor ( ImGuiCol_ChildBg , ImGui : : GetColorU32 ( ImGuiCol_TableHeaderBg ) ) ;
ImGui : : PushStyleVar ( ImGuiStyleVar_ChildRounding , 50 ) ;
ImGui : : PushStyleVar ( ImGuiStyleVar_WindowPadding , scaled ( { 12 , 3 } ) ) ;
if ( ImGui : : BeginChild ( library . link , ImVec2 ( ) , ImGuiChildFlags_Border | ImGuiChildFlags_AutoResizeX | ImGuiChildFlags_AutoResizeY ) ) {
if ( ImGuiExt : : Hyperlink ( hex : : format ( " {}/{} " , library . author , library . name ) . c_str ( ) ) ) {
hex : : openWebpage ( library . link ) ;
}
2023-11-25 13:15:21 +01:00
ImGui : : SetItemTooltip ( " %s " , library . link ) ;
2023-11-25 00:43:03 +01:00
}
ImGui : : EndChild ( ) ;
2021-01-23 00:46:50 +01:00
2023-11-25 00:43:03 +01:00
ImGui : : SameLine ( ) ;
2023-11-25 12:44:20 +01:00
if ( ImGui : : GetCursorPosX ( ) > ( width - 100 _scaled ) )
2023-11-25 00:43:03 +01:00
ImGui : : NewLine ( ) ;
2022-03-04 19:06:29 +01:00
2023-11-25 00:43:03 +01:00
ImGui : : PopStyleColor ( ) ;
ImGui : : PopStyleVar ( 2 ) ;
}
}
ImGuiExt : : EndSubWindow ( ) ;
2023-11-25 12:44:20 +01:00
ImGui : : NewLine ( ) ;
2023-11-25 00:43:03 +01:00
} ;
ImGuiExt : : TextFormattedWrapped ( " ImHex builds on top of the amazing work of a ton of talented library developers without which this project wouldn't stand. " ) ;
2022-03-04 19:06:29 +01:00
ImGui : : NewLine ( ) ;
2023-11-25 00:43:03 +01:00
drawTable ( " ImGui " , ImGuiLibraries ) ;
drawTable ( " External " , ExternalLibraries ) ;
drawTable ( " Third Party " , ThirdPartyLibraries ) ;
2021-12-12 13:35:07 +01:00
}
2021-01-23 00:46:50 +01:00
2023-12-18 11:21:33 +01:00
void ViewAbout : : drawLoadedPlugins ( ) {
const auto & plugins = PluginManager : : getPlugins ( ) ;
ImGui : : PushStyleVar ( ImGuiStyleVar_WindowPadding , ImVec2 ( ) ) ;
ImGuiExt : : BeginSubWindow ( " hex.builtin.view.help.about.plugins " _lang ) ;
ImGui : : PopStyleVar ( ) ;
{
2023-12-31 11:39:24 +01:00
if ( ImGui : : BeginTable ( " plugins " , 4 , ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg | ImGuiTableFlags_SizingFixedFit ) ) {
2023-12-18 11:21:33 +01:00
ImGui : : TableSetupScrollFreeze ( 0 , 1 ) ;
ImGui : : TableSetupColumn ( " hex.builtin.view.help.about.plugins.plugin " _lang ) ;
ImGui : : TableSetupColumn ( " hex.builtin.view.help.about.plugins.author " _lang ) ;
ImGui : : TableSetupColumn ( " hex.builtin.view.help.about.plugins.desc " _lang , ImGuiTableColumnFlags_WidthStretch , 0.5 ) ;
ImGui : : TableSetupColumn ( " ##loaded " , ImGuiTableColumnFlags_WidthFixed , ImGui : : GetTextLineHeight ( ) ) ;
ImGui : : TableHeadersRow ( ) ;
2023-12-23 21:09:41 +01:00
for ( const auto & plugin : plugins ) {
if ( plugin . isLibraryPlugin ( ) )
continue ;
2023-12-18 11:21:33 +01:00
2023-12-31 11:39:24 +01:00
auto features = plugin . getFeatures ( ) ;
2023-12-23 21:09:41 +01:00
ImGui : : TableNextRow ( ) ;
ImGui : : TableNextColumn ( ) ;
2023-12-31 11:39:24 +01:00
bool open = false ;
2024-01-12 23:03:13 +01:00
ImGui : : PushStyleColor ( ImGuiCol_Text , ImGui : : GetColorU32 ( ImGuiCol_Text ) ) ;
2023-12-31 11:39:24 +01:00
if ( features . empty ( ) )
ImGui : : BulletText ( " %s " , plugin . getPluginName ( ) . c_str ( ) ) ;
else
open = ImGui : : TreeNode ( plugin . getPluginName ( ) . c_str ( ) ) ;
ImGui : : PopStyleColor ( ) ;
2023-12-23 21:09:41 +01:00
ImGui : : TableNextColumn ( ) ;
ImGui : : TextUnformatted ( plugin . getPluginAuthor ( ) . c_str ( ) ) ;
ImGui : : TableNextColumn ( ) ;
ImGui : : TextUnformatted ( plugin . getPluginDescription ( ) . c_str ( ) ) ;
ImGui : : TableNextColumn ( ) ;
ImGui : : TextUnformatted ( plugin . isLoaded ( ) ? ICON_VS_CHECK : ICON_VS_CLOSE ) ;
2023-12-31 11:39:24 +01:00
if ( open ) {
for ( const auto & feature : plugin . getFeatures ( ) ) {
ImGui : : TableNextRow ( ) ;
ImGui : : TableNextColumn ( ) ;
ImGuiExt : : TextFormatted ( " {} " , feature . name . c_str ( ) ) ;
ImGui : : TableNextColumn ( ) ;
ImGui : : TableNextColumn ( ) ;
ImGui : : TableNextColumn ( ) ;
ImGui : : TextUnformatted ( feature . enabled ? ICON_VS_CHECK : ICON_VS_CLOSE ) ;
}
ImGui : : TreePop ( ) ;
}
2023-12-18 11:21:33 +01:00
}
ImGui : : EndTable ( ) ;
}
}
ImGuiExt : : EndSubWindow ( ) ;
}
2022-03-04 19:06:29 +01:00
void ViewAbout : : drawPathsPage ( ) {
2023-12-11 21:29:30 +01:00
constexpr static std : : array < std : : pair < const char * , fs : : ImHexPath > , size_t ( fs : : ImHexPath : : END ) > PathTypes = {
2023-11-25 00:43:03 +01:00
{
2024-03-26 19:18:34 +01:00
{ " Patterns " , fs : : ImHexPath : : Patterns } ,
{ " Patterns Includes " , fs : : ImHexPath : : PatternsInclude } ,
{ " Magic " , fs : : ImHexPath : : Magic } ,
{ " Plugins " , fs : : ImHexPath : : Plugins } ,
{ " Yara Patterns " , fs : : ImHexPath : : Yara } ,
{ " Yara Advaned Analysis " , fs : : ImHexPath : : YaraAdvancedAnalysis } ,
{ " Config " , fs : : ImHexPath : : Config } ,
{ " Backups " , fs : : ImHexPath : : Backups } ,
{ " Resources " , fs : : ImHexPath : : Resources } ,
{ " Constants lists " , fs : : ImHexPath : : Constants } ,
{ " Custom encodings " , fs : : ImHexPath : : Encodings } ,
{ " Logs " , fs : : ImHexPath : : Logs } ,
{ " Recent files " , fs : : ImHexPath : : Recent } ,
{ " Scripts " , fs : : ImHexPath : : Scripts } ,
{ " Data inspector scripts " , fs : : ImHexPath : : Inspectors } ,
{ " Themes " , fs : : ImHexPath : : Themes } ,
{ " Native Libraries " , fs : : ImHexPath : : Libraries } ,
{ " Custom data processor nodes " , fs : : ImHexPath : : Nodes } ,
{ " Layouts " , fs : : ImHexPath : : Layouts } ,
{ " Workspaces " , fs : : ImHexPath : : Workspaces } ,
2023-11-25 00:43:03 +01:00
}
} ;
2024-03-26 19:18:34 +01:00
static_assert ( PathTypes . back ( ) . first ! = nullptr , " All path items need to be populated! " ) ;
2021-12-12 13:35:07 +01:00
2023-11-25 00:43:03 +01:00
ImGui : : PushStyleVar ( ImGuiStyleVar_WindowPadding , ImVec2 ( ) ) ;
ImGuiExt : : BeginSubWindow ( " Paths " , ImGui : : GetContentRegionAvail ( ) ) ;
{
if ( ImGui : : BeginTable ( " ##imhex_paths " , 2 , ImGuiTableFlags_ScrollY | ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg | ImGuiTableFlags_SizingFixedFit ) ) {
ImGui : : TableSetupScrollFreeze ( 0 , 1 ) ;
ImGui : : TableSetupColumn ( " Type " ) ;
ImGui : : TableSetupColumn ( " Paths " ) ;
// Draw the table
ImGui : : TableHeadersRow ( ) ;
for ( const auto & [ name , type ] : PathTypes ) {
ImGui : : TableNextRow ( ) ;
ImGui : : TableNextColumn ( ) ;
ImGui : : TextUnformatted ( name ) ;
2021-12-12 13:35:07 +01:00
2023-11-25 00:43:03 +01:00
ImGui : : TableNextColumn ( ) ;
for ( auto & path : fs : : getDefaultPaths ( type , true ) ) {
// Draw hyperlink to paths that exist or red text if they don't
if ( wolv : : io : : fs : : isDirectory ( path ) ) {
if ( ImGuiExt : : Hyperlink ( wolv : : util : : toUTF8String ( path ) . c_str ( ) ) ) {
fs : : openFolderExternal ( path ) ;
}
} else {
ImGuiExt : : TextFormattedColored ( ImGuiExt : : GetCustomColorVec4 ( ImGuiCustomCol_ToolbarRed ) , wolv : : util : : toUTF8String ( path ) ) ;
2023-06-05 13:50:55 +02:00
}
2022-10-13 08:07:46 +02:00
}
}
2021-12-12 13:35:07 +01:00
2023-11-25 00:43:03 +01:00
ImGui : : EndTable ( ) ;
}
2021-12-12 13:35:07 +01:00
}
2023-11-25 00:43:03 +01:00
ImGuiExt : : EndSubWindow ( ) ;
ImGui : : PopStyleVar ( ) ;
2021-12-12 13:35:07 +01:00
}
2023-11-05 19:57:29 +01:00
void ViewAbout : : drawReleaseNotesPage ( ) {
static std : : string releaseTitle ;
static std : : vector < std : : string > releaseNotes ;
2022-01-22 23:11:28 +01:00
2023-11-05 19:57:29 +01:00
// Set up the request to get the release notes the first time the page is opened
AT_FIRST_TIME {
static HttpRequest request ( " GET " , GitHubApiURL + std : : string ( " /releases/tags/v " ) + ImHexApi : : System : : getImHexVersion ( false ) ) ;
2021-01-14 17:01:44 +01:00
2023-12-19 13:10:25 +01:00
m_releaseNoteRequest = request . execute ( ) ;
2023-11-05 19:57:29 +01:00
} ;
2021-01-14 17:01:44 +01:00
2023-11-05 19:57:29 +01:00
// Wait for the request to finish and parse the response
2023-12-19 13:10:25 +01:00
if ( m_releaseNoteRequest . valid ( ) ) {
if ( m_releaseNoteRequest . wait_for ( std : : chrono : : seconds ( 0 ) ) = = std : : future_status : : ready ) {
auto response = m_releaseNoteRequest . get ( ) ;
2023-11-05 19:57:29 +01:00
nlohmann : : json json ;
if ( response . isSuccess ( ) ) {
// A valid response was received, parse it
try {
json = nlohmann : : json : : parse ( response . getData ( ) ) ;
// Get the release title
releaseTitle = json [ " name " ] . get < std : : string > ( ) ;
2021-12-12 13:35:07 +01:00
2023-11-05 19:57:29 +01:00
// Get the release notes and split it into lines
auto body = json [ " body " ] . get < std : : string > ( ) ;
releaseNotes = wolv : : util : : splitString ( body , " \r \n " ) ;
} catch ( std : : exception & e ) {
releaseNotes . push_back ( " ## Error: " + std : : string ( e . what ( ) ) ) ;
2022-01-24 00:45:46 +01:00
}
2023-11-05 19:57:29 +01:00
} else {
// An error occurred, display it
releaseNotes . push_back ( " ## HTTP Error: " + std : : to_string ( response . getStatusCode ( ) ) ) ;
2021-05-23 23:35:04 +02:00
}
2023-11-05 19:57:29 +01:00
} else {
// Draw a spinner while the release notes are loading
2023-12-23 21:09:41 +01:00
ImGuiExt : : TextSpinner ( " hex.ui.common.loading " _lang ) ;
2023-11-05 19:57:29 +01:00
}
}
2021-05-23 23:35:04 +02:00
2023-11-05 19:57:29 +01:00
// Function to handle drawing of a regular text line
static const auto drawRegularLine = [ ] ( const std : : string & line ) {
ImGui : : Bullet ( ) ;
ImGui : : SameLine ( ) ;
// Check if the line contains bold text
auto boldStart = line . find ( " ** " ) ;
if ( boldStart ! = std : : string : : npos ) {
// Find the end of the bold text
auto boldEnd = line . find ( " ** " , boldStart + 2 ) ;
// Draw the line with the bold text highlighted
ImGui : : TextUnformatted ( line . substr ( 0 , boldStart ) . c_str ( ) ) ;
ImGui : : SameLine ( 0 , 0 ) ;
2023-11-16 22:24:06 +01:00
ImGuiExt : : TextFormattedColored ( ImGuiExt : : GetCustomColorVec4 ( ImGuiCustomCol_Highlight ) , " {} " , line . substr ( boldStart + 2 , boldEnd - boldStart - 2 ) . c_str ( ) ) ;
2023-11-05 19:57:29 +01:00
ImGui : : SameLine ( 0 , 0 ) ;
ImGui : : TextUnformatted ( line . substr ( boldEnd + 2 ) . c_str ( ) ) ;
} else {
// Draw the line normally
ImGui : : TextUnformatted ( line . c_str ( ) ) ;
}
} ;
// Draw the release title
if ( ! releaseTitle . empty ( ) ) {
auto title = hex : : format ( " v{}: {} " , ImHexApi : : System : : getImHexVersion ( false ) , releaseTitle ) ;
2023-11-16 22:24:06 +01:00
ImGuiExt : : Header ( title . c_str ( ) , true ) ;
2023-11-05 19:57:29 +01:00
ImGui : : Separator ( ) ;
}
// Draw the release notes and format them using parts of the GitHub Markdown syntax
// This is not a full implementation of the syntax, but it's enough to make the release notes look good.
for ( const auto & line : releaseNotes ) {
if ( line . starts_with ( " ## " ) ) {
// Draw H2 Header
2023-11-16 22:24:06 +01:00
ImGuiExt : : Header ( line . substr ( 3 ) . c_str ( ) ) ;
2023-11-05 19:57:29 +01:00
} else if ( line . starts_with ( " ### " ) ) {
// Draw H3 Header
2023-11-16 22:24:06 +01:00
ImGuiExt : : Header ( line . substr ( 4 ) . c_str ( ) ) ;
2023-11-05 19:57:29 +01:00
} else if ( line . starts_with ( " - " ) ) {
// Draw bullet point
drawRegularLine ( line . substr ( 2 ) ) ;
} else if ( line . starts_with ( " - " ) ) {
// Draw further indented bullet point
ImGui : : Indent ( ) ;
ImGui : : Indent ( ) ;
drawRegularLine ( line . substr ( 6 ) ) ;
ImGui : : Unindent ( ) ;
ImGui : : Unindent ( ) ;
}
}
}
void ViewAbout : : drawCommitHistoryPage ( ) {
struct Commit {
std : : string hash ;
std : : string message ;
std : : string description ;
std : : string author ;
std : : string date ;
std : : string url ;
} ;
static std : : vector < Commit > commits ;
// Set up the request to get the commit history the first time the page is opened
AT_FIRST_TIME {
static HttpRequest request ( " GET " , GitHubApiURL + std : : string ( " /commits?per_page=100 " ) ) ;
2023-12-19 13:10:25 +01:00
m_commitHistoryRequest = request . execute ( ) ;
2023-11-05 19:57:29 +01:00
} ;
// Wait for the request to finish and parse the response
2023-12-19 13:10:25 +01:00
if ( m_commitHistoryRequest . valid ( ) ) {
if ( m_commitHistoryRequest . wait_for ( std : : chrono : : seconds ( 0 ) ) = = std : : future_status : : ready ) {
auto response = m_commitHistoryRequest . get ( ) ;
2023-11-05 19:57:29 +01:00
nlohmann : : json json ;
if ( response . isSuccess ( ) ) {
// A valid response was received, parse it
try {
json = nlohmann : : json : : parse ( response . getData ( ) ) ;
for ( auto & commit : json ) {
const auto message = commit [ " commit " ] [ " message " ] . get < std : : string > ( ) ;
// Split commit title and description. They're separated by two newlines.
const auto messageEnd = message . find ( " \n \n " ) ;
auto commitTitle = messageEnd = = std : : string : : npos ? message : message . substr ( 0 , messageEnd ) ;
auto commitDescription = messageEnd = = std : : string : : npos ? " " : message . substr ( commitTitle . size ( ) + 2 ) ;
auto url = commit [ " html_url " ] . get < std : : string > ( ) ;
auto sha = commit [ " sha " ] . get < std : : string > ( ) ;
auto date = commit [ " commit " ] [ " author " ] [ " date " ] . get < std : : string > ( ) ;
auto author = hex : : format ( " {} <{}> " ,
commit [ " commit " ] [ " author " ] [ " name " ] . get < std : : string > ( ) ,
commit [ " commit " ] [ " author " ] [ " email " ] . get < std : : string > ( )
) ;
// Move the commit data into the list of commits
commits . emplace_back (
std : : move ( sha ) ,
std : : move ( commitTitle ) ,
std : : move ( commitDescription ) ,
std : : move ( author ) ,
std : : move ( date ) ,
std : : move ( url )
) ;
}
} catch ( std : : exception & e ) {
commits . emplace_back (
2023-12-23 21:09:41 +01:00
" hex.ui.common.error " _lang ,
2023-11-05 19:57:29 +01:00
e . what ( ) ,
" " ,
" " ,
" "
) ;
2022-01-24 00:45:46 +01:00
}
2023-11-05 19:57:29 +01:00
} else {
// An error occurred, display it
commits . emplace_back (
2023-12-23 21:09:41 +01:00
" hex.ui.common.error " _lang ,
2023-11-05 19:57:29 +01:00
" HTTP " + std : : to_string ( response . getStatusCode ( ) ) ,
" " ,
" " ,
" "
) ;
2021-12-12 13:35:07 +01:00
}
2023-11-05 19:57:29 +01:00
} else {
// Draw a spinner while the commits are loading
2023-12-23 21:09:41 +01:00
ImGuiExt : : TextSpinner ( " hex.ui.common.loading " _lang ) ;
2023-11-05 19:57:29 +01:00
}
}
2021-05-23 23:35:04 +02:00
2023-11-05 19:57:29 +01:00
// Draw commits table
if ( ! commits . empty ( ) ) {
2023-11-25 00:43:03 +01:00
ImGui : : PushStyleVar ( ImGuiStyleVar_WindowPadding , ImVec2 ( ) ) ;
ImGuiExt : : BeginSubWindow ( " Commits " , ImGui : : GetContentRegionAvail ( ) ) ;
2023-12-02 11:09:32 +01:00
ImGui : : PopStyleVar ( ) ;
2023-11-25 00:43:03 +01:00
{
if ( ImGui : : BeginTable ( " ##commits " , 2 , ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg | ImGuiTableFlags_SizingFixedFit | ImGuiTableFlags_ScrollY ) ) {
// Draw commits
for ( const auto & commit : commits ) {
ImGui : : PushID ( commit . hash . c_str ( ) ) ;
ImGui : : TableNextRow ( ) ;
// Draw hover tooltip
ImGui : : TableNextColumn ( ) ;
if ( ImGui : : Selectable ( " ##commit " , false , ImGuiSelectableFlags_SpanAllColumns ) ) {
hex : : openWebpage ( commit . url ) ;
}
2023-11-05 19:57:29 +01:00
2023-11-25 00:43:03 +01:00
if ( ImGui : : IsItemHovered ( ) ) {
if ( ImGui : : BeginTooltip ( ) ) {
// Draw author and commit date
ImGuiExt : : TextFormattedColored ( ImGuiExt : : GetCustomColorVec4 ( ImGuiCustomCol_Highlight ) , " {} " , commit . author ) ;
ImGui : : SameLine ( ) ;
ImGuiExt : : TextFormatted ( " @ {} " , commit . date . c_str ( ) ) ;
// Draw description if there is one
if ( ! commit . description . empty ( ) ) {
ImGui : : Separator ( ) ;
ImGuiExt : : TextFormatted ( " {} " , commit . description ) ;
}
2020-11-14 21:16:03 +01:00
2023-11-25 00:43:03 +01:00
ImGui : : EndTooltip ( ) ;
2023-11-05 19:57:29 +01:00
}
}
2023-11-25 00:43:03 +01:00
// Draw commit hash
ImGui : : SameLine ( 0 , 0 ) ;
ImGuiExt : : TextFormattedColored ( ImGuiExt : : GetCustomColorVec4 ( ImGuiCustomCol_Highlight ) , " {} " , commit . hash . substr ( 0 , 7 ) ) ;
2023-11-05 19:57:29 +01:00
2023-11-25 00:43:03 +01:00
// Draw the commit message
ImGui : : TableNextColumn ( ) ;
2023-11-05 19:57:29 +01:00
2023-11-25 00:43:03 +01:00
const ImColor color = [ & ] {
if ( commit . hash = = ImHexApi : : System : : getCommitHash ( true ) )
return ImGui : : GetStyleColorVec4 ( ImGuiCol_HeaderActive ) ;
else
return ImGui : : GetStyleColorVec4 ( ImGuiCol_Text ) ;
} ( ) ;
ImGuiExt : : TextFormattedColored ( color , commit . message ) ;
2023-11-05 19:57:29 +01:00
2023-11-25 00:43:03 +01:00
ImGui : : PopID ( ) ;
}
2023-11-05 19:57:29 +01:00
2023-11-25 00:43:03 +01:00
ImGui : : EndTable ( ) ;
2021-12-12 13:35:07 +01:00
}
2023-11-05 19:57:29 +01:00
}
2023-11-25 00:43:03 +01:00
ImGuiExt : : EndSubWindow ( ) ;
2023-11-05 19:57:29 +01:00
}
}
void ViewAbout : : drawLicensePage ( ) {
2024-05-03 21:38:35 +02:00
const auto indentation = 50 _scaled ;
ImGui : : Indent ( indentation ) ;
2023-11-16 22:24:06 +01:00
ImGuiExt : : TextFormattedWrapped ( " {} " , romfs : : get ( " licenses/LICENSE " ) . string ( ) ) ;
2024-05-03 21:38:35 +02:00
ImGui : : Unindent ( indentation ) ;
2023-11-05 19:57:29 +01:00
}
void ViewAbout : : drawAboutPopup ( ) {
struct Tab {
using Function = void ( ViewAbout : : * ) ( ) ;
const char * unlocalizedName ;
Function function ;
} ;
constexpr std : : array Tabs = {
Tab { " ImHex " , & ViewAbout : : drawAboutMainPage } ,
Tab { " hex.builtin.view.help.about.contributor " , & ViewAbout : : drawContributorPage } ,
Tab { " hex.builtin.view.help.about.libs " , & ViewAbout : : drawLibraryCreditsPage } ,
2023-12-18 11:21:33 +01:00
Tab { " hex.builtin.view.help.about.plugins " , & ViewAbout : : drawLoadedPlugins } ,
2023-11-05 19:57:29 +01:00
Tab { " hex.builtin.view.help.about.paths " , & ViewAbout : : drawPathsPage } ,
Tab { " hex.builtin.view.help.about.release_notes " , & ViewAbout : : drawReleaseNotesPage } ,
Tab { " hex.builtin.view.help.about.commits " , & ViewAbout : : drawCommitHistoryPage } ,
Tab { " hex.builtin.view.help.about.license " , & ViewAbout : : drawLicensePage } ,
} ;
2023-11-21 13:47:50 +01:00
// Allow the window to be closed by pressing ESC
2024-04-12 22:56:10 +02:00
if ( ImGui : : IsKeyDown ( ImGuiKey_Escape ) )
2023-11-21 13:47:50 +01:00
ImGui : : CloseCurrentPopup ( ) ;
2023-11-05 19:57:29 +01:00
2023-11-21 13:47:50 +01:00
if ( ImGui : : BeginTabBar ( " about_tab_bar " ) ) {
// Draw all tabs
for ( const auto & [ unlocalizedName , function ] : Tabs ) {
2023-11-21 14:38:01 +01:00
if ( ImGui : : BeginTabItem ( Lang ( unlocalizedName ) ) ) {
2023-11-21 13:47:50 +01:00
ImGui : : NewLine ( ) ;
2023-11-05 19:57:29 +01:00
2023-11-21 13:47:50 +01:00
if ( ImGui : : BeginChild ( 1 ) ) {
( this - > * function ) ( ) ;
2022-01-24 00:45:46 +01:00
}
2023-11-21 13:47:50 +01:00
ImGui : : EndChild ( ) ;
2022-01-22 23:11:28 +01:00
2023-11-21 13:47:50 +01:00
ImGui : : EndTabItem ( ) ;
}
2021-12-12 13:35:07 +01:00
}
2020-11-30 21:44:40 +01:00
2023-11-21 13:47:50 +01:00
ImGui : : EndTabBar ( ) ;
2020-11-14 21:16:03 +01:00
}
}
2022-03-04 19:06:29 +01:00
void ViewAbout : : drawContent ( ) {
2020-11-17 13:59:16 +01:00
this - > drawAboutPopup ( ) ;
2020-11-14 21:16:03 +01:00
}
2021-08-25 13:54:59 -04:00
}